您好,登錄后才能下訂單哦!
本篇文章給大家分享的是有關如何在java 8項目中自定義collector,小編覺得挺實用的,因此分享給大家學習,希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。
需求:
將 一個容器List<Bean> 按照一定的字段進行分組,分組過后的值為特定的BEAN 里面的屬性例如:
假定有這樣一個Bean
public class SubjectOberser{ private String subjectKey; private AbstractObserver abstractObserver; ...geter seter 方法... }
我們需要按照 subjectKey 進行分組,分組過后的內容 應該為這樣一個容器Map<String,List<AbstractObserver>>
map 中的key,為SubjectOberser 屬性的subjectKey,值為List<AbstractObserver>
實現過程
首先來看看collector 的接口定義
public interface Collector<T, A, R> { Supplier<A> supplier(); BiConsumer<A, T> accumulator(); Function<A, R> finisher(); BinaryOperator<A> combiner(); Set<Characteristics> characteristics(); }
類型 T ,是在容器里面元素的類型
類型 A ,是accumulator 返回的類型,即是累加器的返回類型
類型 R ,是最終結果的類型
supplier 方法返回的結果必須為一個空的Supplier,也就是一個空的無參函數(簽名就是這樣的 ()->{}),在調用的時候它會創建一個空的累加器(accumulator)實例,供數據收集的時候使用,很明顯如果按照我們的需求試下你自己collector 這里應該返回一個 () -> new HashMap<>()
,一個Map 來收集結果
accumulator 方法返回歸約操作的函數(簽名是這樣的 (a,b)->void ),當遍歷到流中第n個元素時,這個函數執行時會有兩個參數:保存歸約結果的累加器(已 收集了流中的前n-1個項目),還有第n個元素本身。簽名也展示該函數是void,因為該操作是在原來的容器里面進行更新的,所以返回的是void 類型。按照需求的中的實現應該是是這樣的:
public BiConsumer<Map<String, List<AbstractObserver>>, SubjectObserver> accumulator() { return (Map<String, List<AbstractObserver>> acc, SubjectObserver v) -> { if (acc.containsKey(v.getSubjectKey())){ acc.get(v.getSubjectKey()).add(v.getAbstractObserver()); }else{ List<AbstractObserver> l = new ArrayList<>(); l.add(v.getAbstractObserver()); acc.put(v.getSubjectKey(),l); } }; }
這里的邏輯就是if else 邏輯判斷就是,這個key ,在map 中是否存在,如果不存在,那么我們需要給他new一個list 的實例,不然我的的數據沒有地方存儲
finisher 可從名字看出方法累積過程的最后要調用的一個函數,以便將累加器對象轉換為整個集合操作的最終結果。通常來說累加器的類型也是返回的結果的類型,那么就返回identity 就可以了,如果不是的話,就行自行轉換了。在當前需求的情況下我們的累加器和返回結果的類型是一致的,所以這里的實現是這樣的:
public Function<Map<String, List<AbstractObserver>>, Map<String, List<AbstractObserver>>> finisher(){ return Function.identity(); }
combiner 方法是將兩個累加的結果進行一個合并的過程,當然這個過程并不是每一個collector 都會調用得到(后面會講到)
按照我們的需求,只需要將兩個累加器的,中間結果合并成為一個結果即可,所以是現實這樣的:
public BinaryOperator<Map<String, List<AbstractObserver>>> combiner() { return ((Map<String, List<AbstractObserver>> map1, Map<String, List<AbstractObserver>> map2) -> { map1.putAll(map2); return map1; }); }
characteristics 該方法返回一個 Characteristics 的集合,它有如下值可選
UNORDERED—— 歸約結果不受流中項目的遍歷和累積順序的影響。
CONCURRENT—— accumulator函數可以從多個線程同時調用,且該收集器可以并行執行。如果收集器沒有標為UNORDERED,那 它僅在用于用于無序數據源時才可以并行歸約。
IDENTITY_ FINISH—— 這表明完成器方法返回的函數是一個不改變的函數,這種情況下,累加器對象將會直接用作合并過程 的最終結果。
public Set<Characteristics> characteristics() { return Collections.unmodifiableSet(EnumSet.of(Characteristics.IDENTITY_FINISH)); }
最終collector 代碼合在一起就是:
public class MyCollector implements Collector<SubjectObserver, Map<String, List<AbstractObserver>>, Map<String, List<AbstractObserver>>> { @Override public Supplier<Map<String, List<AbstractObserver>>> supplier() { return () -> new HashMap<>(); } @Override public BiConsumer<Map<String, List<AbstractObserver>>, SubjectObserver> accumulator() { return (Map<String, List<AbstractObserver>> acc, SubjectObserver v) -> { if (acc.containsKey(v.getSubjectKey())) { acc.get(v.getSubjectKey()).add(v.getAbstractObserver()); } else { List<AbstractObserver> l = new ArrayList<>(); l.add(v.getAbstractObserver()); acc.put(v.getSubjectKey(), l); } }; } @Override public BinaryOperator<Map<String, List<AbstractObserver>>> combiner() { return ((Map<String, List<AbstractObserver>> map1, Map<String, List<AbstractObserver>> map2) -> { map1.putAll(map2); return map1; }); } @Override public Function<Map<String, List<AbstractObserver>>, Map<String, List<AbstractObserver>>> finisher() { return Function.identity(); } @Override public Set<Characteristics> characteristics() { return Collections.unmodifiableSet(EnumSet.of(Characteristics.IDENTITY_FINISH)); } }
調用的過程就是:
public static Map<String, List<AbstractObserver>> initObjectMap() { ClassScaner classScaner = new ClassScaner(); Set<Class> set = classScaner.doScan("com.souche.datacenter.observer"); return set .stream() .filter(aClass -> SubjectAnnotationResolver.getAnnotationSubjectName(aClass) != null) .map(aClass -> { String subjectKey = SubjectAnnotationResolver.getAnnotationSubjectName(aClass); AbstractObserver abstractObserver = getBeanByClassName(aClass.getSimpleName()); return new SubjectObserver(subjectKey, abstractObserver); }).collect(new MyCollector()); }
直接在使用的地方直接new MyCollector
就可以了
以上就是如何在java 8項目中自定義collector,小編相信有部分知識點可能是我們日常工作會見到或用到的。希望你能通過這篇文章學到更多知識。更多詳情敬請關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。