java 8如何自定义收集器(collector)详解
发布时间 - 2026-01-11 02:05:43 点击率:次需求:

将 一个容器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 就可以了
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对的支持。
# java8
# collector
# java
# 垃圾收集器
# Java9中新增的Collector收集器
# Java8 Stream Collectors收集器使用方法解析
# java收集器Collector案例汇总
# 累加器
# 是这样
# 是一个
# 是在
# 遍历
# 这样一个
# 值为
# 就可以
# 的是
# 情况下
# 收集器
# 也就
# 多个
# 将会
# 给他
# 只需
# 你自己
# 就行
# 不受
# 不存在
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
音响网站制作视频教程,隆霸音响官方网站?
Laravel Asset编译怎么配置_Laravel Vite前端构建工具使用
百度输入法全感官ai怎么关 百度输入法全感官皮肤关闭
百度浏览器如何管理插件 百度浏览器插件管理方法
JavaScript如何实现继承_有哪些常用方法
HTML 中动态设置元素 name 属性的正确语法详解
java中使用zxing批量生成二维码立牌
详解jQuery中的事件
香港服务器建站指南:外贸独立站搭建与跨境电商配置流程
详解CentOS6.5 安装 MySQL5.1.71的方法
Laravel集合Collection怎么用_Laravel集合常用函数详解
合肥制作网站的公司有哪些,合肥聚美网络科技有限公司介绍?
高端智能建站公司优选:品牌定制与SEO优化一站式服务
Laravel PHP版本要求一览_Laravel各版本环境要求对照
JS实现鼠标移上去显示图片或微信二维码
Win11怎么修改DNS服务器 Win11设置DNS加速网络【指南】
如何解决hover在ie6中的兼容性问题
Laravel如何自定义错误页面(404, 500)?(代码示例)
javascript基本数据类型及类型检测常用方法小结
HTML透明颜色代码怎么让图片透明_给img元素加透明色的技巧【方法】
Win11摄像头无法使用怎么办_Win11相机隐私权限开启教程【详解】
Laravel如何使用Gate和Policy进行权限控制_Laravel权限判定与策略规则配置
如何快速生成ASP一键建站模板并优化安全性?
如何用低价快速搭建高质量网站?
如何在阿里云服务器自主搭建网站?
如何快速上传建站程序避免常见错误?
如何正确下载安装西数主机建站助手?
Laravel怎么生成URL_Laravel路由命名与URL生成函数详解
微信小程序 HTTPS报错整理常见问题及解决方案
java获取注册ip实例
电商网站制作多少钱一个,电子商务公司的网站制作费用计入什么科目?
在线ppt制作网站有哪些软件,如何把网页的内容做成ppt?
利用JavaScript实现拖拽改变元素大小
电商网站制作价格怎么算,网上拍卖流程以及规则?
Laravel如何编写单元测试和功能测试?(PHPUnit示例)
制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?
如何在沈阳梯子盘古建站优化SEO排名与功能模块?
Laravel如何使用.env文件管理环境变量?(最佳实践)
Internet Explorer官网直接进入 IE浏览器在线体验版网址
如何在IIS7中新建站点?详细步骤解析
Laravel如何升级到最新的版本_Laravel版本升级流程与兼容性处理
深入理解Android中的xmlns:tools属性
哪家制作企业网站好,开办像阿里巴巴那样的网络公司和网站要怎么做?
Laravel Debugbar怎么安装_Laravel调试工具栏配置指南
Laravel如何集成Inertia.js与Vue/React?(安装配置)
如何自己制作一个网站链接,如何制作一个企业网站,建设网站的基本步骤有哪些?
西安专业网站制作公司有哪些,陕西省建行官方网站?
php 三元运算符实例详细介绍
Laravel Seeder怎么填充数据_Laravel数据库填充器的使用方法与技巧
常州企业网站制作公司,全国继续教育网怎么登录?

