我是靠谱客的博主 孝顺丝袜,这篇文章主要介绍Java8 Stream 自定义收集器Collector,现在分享给大家,希望可以做个参考。

在之前的例子中,我们都是使用Collectors的静态方法提供的CollectorImpl,为接口Collector<T, A, R>的一个实现类,为了自定义我们自己的Collector,先来分析一下Collector接口。

一、分析接口Collector

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
/**  * @param <T> 要收集的元素的泛型  * @param <A> 累加器容器的类型,  * @param <R> 收集操作得到的对象类型  * @author futao  * @date 2020/9/24  */ public class MyCustomCollector<T, A, R> implements Collector<T, A, R> {     /**      * 创建一个接收结果的可变容器      *      * @return a function which returns a new, mutable result container      */     @Override     public Supplier<A> supplier() {         return null;     }     /**      * 将流中的元素放入可变容器中的逻辑, 方法      *      * @return a function which folds a value into a mutable result container      */     @Override     public BiConsumer<A, T> accumulator() {         return null;     }     /**      * 组合结果,当流被拆分成多个部分时,需要将多个结果合并。      *      * @return a function which combines two partial results into a combined      * result      */     @Override     public BinaryOperator<A> combiner() {         return null;     }     /**      * 最后调用:在遍历完流后将结果容器A转换为最终结果R      *      * @return a function which transforms the intermediate result to the final      * result      */     @Override     public Function<A, R> finisher() {         return null;     }     /**      * 返回一个描述收集器特征的不可变集合,用于告诉收集器时可以进行哪些优化,如并行化      *      * @return an immutable set of collector characteristics      */     @Override     public Set<Characteristics> characteristics() {         return null;     } }
  • 描述收集器的特征: enum Characteristics

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
enum Characteristics {     /**      * 意味着结果容器支持多线程并发操作accumulator()方法向容器中放入元素。      * 如果收集器没有被同时标记为无序的`UNORDERED`,则该特征只在数据源为无序(如set)时才有效      */     CONCURRENT,     /**      * 规约操作不保证集合中元素的顺序      * 如结果容器为Set的场景下      */     UNORDERED,     /**      * 表明`完成方法finisher`是一个恒等式,可以被忽略      * 累加器的结果combiner A 将会作为最终的结果R      * 这也要求,直接将A转换成R是安全的      */     IDENTITY_FINISH }
  • Collectors中常见的Characteristics组合

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
    static final Set<Collector.Characteristics> CH_CONCURRENT_ID             = Collections.unmodifiableSet(EnumSet.of(Collector.Characteristics.CONCURRENT,                                                      Collector.Characteristics.UNORDERED,                                                      Collector.Characteristics.IDENTITY_FINISH));     static final Set<Collector.Characteristics> CH_CONCURRENT_NOID             = Collections.unmodifiableSet(EnumSet.of(Collector.Characteristics.CONCURRENT,                                                      Collector.Characteristics.UNORDERED));     static final Set<Collector.Characteristics> CH_ID             = Collections.unmodifiableSet(EnumSet.of(Collector.Characteristics.IDENTITY_FINISH));     static final Set<Collector.Characteristics> CH_UNORDERED_ID             = Collections.unmodifiableSet(EnumSet.of(Collector.Characteristics.UNORDERED,                                                      Collector.Characteristics.IDENTITY_FINISH));     static final Set<Collector.Characteristics> CH_NOID = Collections.emptySet();
  • Collector执行规约过程

图片来源:《java8 in action》

二、自定义一个功能与Collectors.toList()一致的Collector

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
/**  * 自定义收集器  *  * @author futao  * @date 2020/9/24  */ public class MyCollectors {     private MyCollectors() {     }     /**      * 描述:将流中的元素转换成List<T>输出      *      * Collector三个泛型:      *                  <入参泛型类型,      *                  中间结果容器类型(在这个例子中为List<T>),      *                  最终结果容器类型(这个例子中也是List<T>)>      * @param <T>      */     public static class ToList<T> implements Collector<T, List<T>, List<T>> {         /**          * 创建一个接收结果的可变容器          * 即:创建一个List<T>对象的方法          *          * @return          */         @Override         public Supplier<List<T>> supplier() {             return ArrayList::new;         }         /**          * 将流中的元素放入可变容器中的方法          *          * @return          */         @Override         public BiConsumer<List<T>, T> accumulator() {             // return (list, item) -> list.add(item);             return List::add;         }         /**          * 组合结果,当流被拆分成多个部分时,需要将多个结果合并。          *          * @return          */         @Override         public BinaryOperator<List<T>> combiner() {             return (list1, list2) -> {                 list1.addAll(list2);                 return list1;             };         }         /**          * 最后调用:在遍历完流后将结果容器A转换为最终结果R          * 在该例子中,combiner结果可作为最终结果,所以返回一个恒等式          *          * @return          */         @Override         public Function<List<T>, List<T>> finisher() {             // return x -> x;             return Function.identity();         }         /**          * 返回一个描述收集器特征的不可变集合          * 该例子中可用的特性是:          *                  finisher可跳过,直接将combiner结果返回。          *                  需要保证有序          *                  不可并发          *          * @return          */         @Override         public Set<Characteristics> characteristics() {             return Collections.unmodifiableSet(EnumSet.of(Characteristics.IDENTITY_FINISH));         }     } }
  • 测试

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
public static void main(String[] args) {     Apple chinaApple = new Apple(10, "中国");     Apple usApple = new Apple(20, "米国");     Apple koreaApple = new Apple(30, "韩国");     Apple japanApple = new Apple(40, "日本");     List<Apple> collect = Stream.of(chinaApple, usApple, koreaApple, japanApple)             // 使用自定义的收集器             .collect(new MyCollectors.ToList<>());     System.out.println(JSON.toJSONString(collect, true)); }
  • 结果

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
[  {   "country":"中国",   "weight":10  },  {   "country":"米国",   "weight":20  },  {   "country":"韩国",   "weight":30  },  {   "country":"日本",   "weight":40  } ]

三、 当finisher为恒等式时的简便写法

复制代码
1
2
3
4
5
6
    List<Apple> simpleListApples = Stream.of(chinaApple, usApple, koreaApple, japanApple)             // 对于finisher为恒等式,可以用这种简单的写法             .<List<Apple>>collect(ArrayList::new, List::add, List::addAll);     System.out.println(JSON.toJSONString(simpleListApples, true));
  • 使用这种方式定义的Collector收集器默认的特性是IDENTITY_FINISHCONCURRENT组合,当数据源为无序集合时,CONCURRENT才会体现出作用。

欢迎在评论区留下你看文章时的思考,及时说出,有助于加深记忆和理解,还能和像你一样也喜欢这个话题的读者相遇~

最后

以上就是孝顺丝袜最近收集整理的关于Java8 Stream 自定义收集器Collector的全部内容,更多相关Java8内容请搜索靠谱客的其他文章。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(47)

评论列表共有 0 条评论

立即
投稿
返回
顶部