中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

如何在java 8項目中自定義collector

發布時間:2020-11-25 15:31:48 來源:億速云 閱讀:155 作者:Leah 欄目:編程語言

本篇文章給大家分享的是有關如何在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,小編相信有部分知識點可能是我們日常工作會見到或用到的。希望你能通過這篇文章學到更多知識。更多詳情敬請關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

临西县| 普格县| 凤凰县| 棋牌| 波密县| 方正县| 武汉市| 屯留县| 台中县| 龙门县| 遵化市| 抚州市| 治多县| 侯马市| 上饶市| 唐河县| 益阳市| 公安县| 精河县| 波密县| 渝中区| 晋城| 和田县| 手游| 治县。| 神农架林区| 即墨市| 抚远县| 新乐市| 苍梧县| 曲沃县| 济宁市| 苍南县| 江孜县| 磐石市| 临泽县| 连城县| 琼结县| 河南省| 定日县| 桓仁|