Java8之Stream-强大的collect操作
admin
2024-02-09 20:12:48
0

collect应该说是Stream中最强大的终端操作了,使用其几乎能得到你想要的任意数据的聚合,下面好好分析该工具的用法.


在Stream接口中有如下两个方法

   R collect(Supplier supplier,BiConsumer accumulator,BiConsumer combiner); R collect(Collector collector);复制代码

很明显第一种相当于简易实现版本,第二种为高级用法.更多更复杂的操作都封装到Collector接口中,并提供一些静态方法供使用者调用.下面逐一分析.

简易调用形式

简易调用形式就是第一种接口,接口如下

   R collect(Supplier supplier,BiConsumer accumulator,BiConsumer combiner);复制代码

调用方式如下,很明显第一个参数supplier为结果存放容器,第二个参数accumulator为结果如何添加到容器的操作,第三个参数combiner则为多个容器的聚合策略.

String concat = stringStream.collect(StringBuilder::new, StringBuilder::append,StringBuilder::append).toString();
//等价于上面,这样看起来应该更加清晰
String concat = stringStream.collect(() -> new StringBuilder(),(l, x) -> l.append(x), (r1, r2) -> r1.append(r2)).toString();复制代码

那么换一种,我想对一个List收集结果总和,按照Collect的要求,首先需要容器sum,然后添加操作 sum+x,聚合操作,sum1+sum2,那么就很容易写出来了,看完下面代码后好好体会下,然后再看高级用法.当然用sum方法收集是最佳解决方案,这里只是提供一种示例应用.

// 由于基本类型都是不可变类型,所以这里用数组当做容器
final Integer[] integers = Lists.newArrayList(1, 2, 3, 4, 5).stream().collect(() -> new Integer[]{0}, (a, x) -> a[0] += x, (a1, a2) -> a1[0] += a2[0]);复制代码

那么再换一种,有一个Person类,其拥有type与name两个属性,那么使用collect把他收集到Map集合中,其中键为type,值为person的集合.如下代码所示,看明白了相信就掌握了该方法.

   Lists.newArrayList().stream().collect(() -> new HashMap>(),(h, x) -> {List value = h.getOrDefault(x.getType(), Lists.newArrayList());value.add(x);h.put(x.getType(), value);},HashMap::putAll);复制代码

Collector高级调用

Collector接口是使得collect操作强大的终极武器,对于绝大部分操作可以分解为旗下主要步骤,提供初始容器->加入元素到容器->并发下多容器聚合->对聚合后结果进行操作,同时Collector接口又提供了of静态方法帮助你最大化的定制自己的操作,官方也提供了Collectors这个类封装了大部分的常用收集操作.
另外CollectorImplCollector的实现类,因为接口不可实例化,这里主要完成实例化操作.

    //初始容器Supplier supplier();//加入到容器操作BiConsumer accumulator();//多容器聚合操作BinaryOperator combiner();//聚合后的结果操作Function finisher();//操作中便于优化的状态字段Set characteristics();复制代码

Collectors的方法封装

Collectors作为官方提供的收集工具类,那么其很多操作都具有参考性质,能帮助我们更加理解Collector接口,万变不离其宗,最终只是上面五个函数接口的混合操作,下面来分析下官方是如何使用这几个接口的.

toList()

容器: ArrayList::new
加入容器操作: List::add
多容器合并: left.addAll(right); return left;
聚合后的结果操作: 这里直接返回,因此无该操作,默认为castingIdentity()
优化操作状态字段: CH_ID
这样看起来很简单,那么对于Map,Set等操作都是类似的实现.

   public static Collector> toList() {return new CollectorImpl<>((Supplier>) ArrayList::new, List::add,(left, right) -> { left.addAll(right); return left; },CH_ID);}复制代码

joining()

容器: StringBuilder::new
加入容器操作: StringBuilder::append
多容器合并: r1.append(r2); return r1;
聚合后的结果操作: StringBuilder::toString
优化操作状态字段: CH_NOID

    public static Collector joining() {return new CollectorImpl(StringBuilder::new, StringBuilder::append,(r1, r2) -> { r1.append(r2); return r1; },StringBuilder::toString, CH_NOID);}复制代码

下面来个复杂的

groupingBy()

groupingBytoMap的一种高级方式,弥补了toMap对值无法提供多元化的收集操作,比如对于返回Map>这样的形式toMap就不是那么顺手,那么groupingBy的重点就是对Key和Value值的处理封装.分析如下代码,其中classifier是对key值的处理,mapFactory则是指定Map的容器具体类型,downstream为对Value的收集操作,具体代码这里不做分析,无非是把值一个一个的put进指定容器.

   public static >Collector groupingBy(Function classifier,Supplier mapFactory,Collector downstream) {.......}复制代码

对于之前用原生collect方法做的收集操作那么就可以很容易改写为groupBy形式

//原生形式Lists.newArrayList().stream().collect(() -> new HashMap>(),(h, x) -> {List value = h.getOrDefault(x.getType(), Lists.newArrayList());value.add(x);h.put(x.getType(), value);},HashMap::putAll);
//groupBy形式
Lists.newArrayList().stream().collect(Collectors.groupingBy(Person::getType, HashMap::new, Collectors.toList()));
//因为对值有了操作,因此我可以更加灵活的对值进行转换
Lists.newArrayList().stream().collect(Collectors.groupingBy(Person::getType, HashMap::new, Collectors.mapping(Person::getName,Collectors.toSet())));复制代码

reducing()

reducing是针对单个值的收集,其返回结果不是集合家族的类型,而是单一的实体类T
容器: boxSupplier(identity),这里包裹用的是一个长度为1的Object[]数组,至于原因自然是不可变类型的锅
加入容器操作: a[0] = op.apply(a[0], t)
多容器合并: a[0] = op.apply(a[0], b[0]); return a;
聚合后的结果操作: 结果自然是Object[0]所包裹的数据a -> a[0]
优化操作状态字段: CH_NOID
那么看到这里困惑是不是有一种恍然大悟的感觉,反正我是有的.

  public static  Collectorreducing(T identity, BinaryOperator op) {return new CollectorImpl<>(boxSupplier(identity),(a, t) -> { a[0] = op.apply(a[0], t); },(a, b) -> { a[0] = op.apply(a[0], b[0]); return a; },a -> a[0],CH_NOID);}复制代码

那么接下来就是对之前Collect的一些操作的改造

//原生操作
final Integer[] integers = Lists.newArrayList(1, 2, 3, 4, 5).stream().collect(() -> new Integer[]{0}, (a, x) -> a[0] += x, (a1, a2) -> a1[0] += a2[0]);
//reducing操作
final Integer collect = Lists.newArrayList(1, 2, 3, 4, 5).stream().collect(Collectors.reducing(0, Integer::sum));    
//当然Stream也提供了reduce操作
final Integer collect = Lists.newArrayList(1, 2, 3, 4, 5).stream().reduce(0, Integer::sum)复制代码

可能遇到的问题

记录下生产中使用该工具遇到的一些小错误

toMap所产生的异常

toMap的操作主要如下代码,异常来自两个方面

  1. 操作调用的是map.merge方法,该方法遇到value为null的情况会报npe,即使你使用的是hashMap可以接受null值,也照样报.搞不懂这里为什么这样设计.
  2. 未指定冲突合并策略,也就是第三个参数BinaryOperator mergeFunction时遇到重复的key会直接抛IllegalStateException,因此需要注意.

总结

到此对于collect的操作应该就很清晰了,希望通过这些例子能掌握核心,也就是Collector接口中那几个函数的作用,希望对你有帮助.

相关内容

热门资讯

鲁泰纺织股份有限公司 2025... 证券代码:000726 200726 证券简称:鲁泰A 鲁泰B 公告编号:2026-002 债券代码...
特朗普再发关税威胁!加拿大回应 【导读】特朗普威胁对加拿大征收100%关税,加总理回应 中国基金报记者 李智 一起来关注下海外资讯。...
被特朗普暴击的马克龙,又放弃威... 被特朗普暴击的马克龙,现在又放弃威胁中方了,突然变脸希望中国投资欧洲,表明西方国家已经开始觉醒。 最...
原创 异... 在浩瀚的星空下,每一颗星星都承载着一个关于爱情的故事。异地恋,这个让无数情侣夜不能寐的话题,究竟能否...
小儿癫痫认知康复护理实践 小儿癫痫认知康复护理方法,涵盖康复认知护理、家庭认知康复护理辅助、护理宣教等手段,以有效促进患儿康复...
钻戒回收价腰斩,银保温杯身价翻... 谁能想到,贵金属市场正在上演一出荒诞大戏 就在上个月,朋友小琳还满心欢喜地戴着新买的钻戒。那是她存...
大反转!美联储,突发! 【导读】预测市场交易员押注里克·里德尔将担任美联储主席 中国基金报记者 泰勒 大家好,简单关注一下美...
特朗普敲定美联储主席人选进入倒... 来源:环球市场播报 数月以来,债券投资者一直押注美联储主席杰罗姆・鲍威尔的继任者将会推动降息。如今,...
上市首日频频大涨,网上冻结资金... 转自:证券时报网 上市首日频频大涨,网上冻结资金刷新纪录!北交所新股火热 作者:钟恬 2026年开年...
北京“东西海朝”二手房靠什么成... 尽管北京楼市整体进入调整期,但市场分化显著,部分二手房源仍展现出强劲的韧性。综合多家中介机构提供的成...
成都一小学强制教师参与年会表演... 1月25日,成都市锦江区教育局发布情况通报: 近日,接群众投诉成都师范附属小学慧源校区存在“强制教师...
时隔半年,德国议员再提撤回存放... 据CCTV国际时讯援引德国媒体1月23日报道,德国议员施特拉克-齐默尔曼呼吁德国联邦政府将存放在美国...
双汇、唐人神、雪川上新;牧原股... 双汇中式卤味“卤福斋”上市 牧原股份最快2月份在香港上市 正大2026将在中国养小棚虾 1.双汇中式...
上交所对出版传媒、关联方宽甸新... 上交所发布关于对北方联合出版传媒(集团)股份有限公司、关联方宽甸满族自治县新华书店有限公司及有关责任...
原创 2... 2025年中国区域经济的最新画卷徐徐展开,21个省份的经济成绩单已新鲜出炉。透过这些跃动的数字,我们...
原创 马... 近日一则消息震惊全球财富圈,据《财联社》等媒体1月20日的报道:马斯克凭借旗下xAI公司的最新一轮融...
白羽鸡海外引种“断供” 国产种... 图虫创意/供图 证券时报记者 黄翔 2026年伊始,一场由海外禽流感引发的产业震荡,波及我国白羽肉...
Vitalik 谈为什么去中心... 来源:市场资讯 (来源:吴说) 在吴说主持的推特 space 中,Vitalik 表示,很多 Cry...
2026年跨境电商财税合规品牌... 随着全球数字经济深化与各国税务监管趋严,跨境电商行业已步入“合规驱动增长”的新阶段。2026年,财税...
海目星涨8.71%,成交额11... 来源:新浪证券-红岸工作室 1月23日,海目星涨8.71%,成交额11.34亿元,换手率6.89%,...