Java 8 Stream 实现按 ID 和日期分组并合并同组订单金额
发布时间 - 2026-01-01 00:00:00 点击率:次本文介绍如何使用 java 8 stream 的 `collectors.tomap` 配合自定义键(如 `record idanddate`)与合并函数,高效地对订单列表按 `id` 和 `date` 两字段分组,并原地或不可变地聚合 `amount`,最终直接获得合并后的 `order` 列表,避免嵌套 `groupingby` 和中间 `map
在 Java 8 Stream 中,若需按多个字段联合分组(如 id + date)并合并同组元素(如累加 amount),最简洁、高效的方式并非嵌套 groupingBy,而是使用 Collectors.toMap —— 它天然支持键冲突时的自定义合并逻辑(即 mergeFunction),且可直接产出 Map
✅ 推荐方案:toMap + 复合键 record
首先,定义一个不可变、可作为 Map 键的轻量级复合键类型(Java 14+ 推荐用 record,兼容性好且语义清晰):
record IdAndDate(Integer id, LocalDate date) {}接着,使用 Collectors.toMap 构建映射:
- keyMapper:将每个 Order 映射为 IdAndDate(id, date);
- valueMapper:直接保留原 Order 对象(Function.identity());
-
mergeFunction:当键重复时,调用 Order::combine 合并金额(注意:此方法当前为就地修改
)。
完整代码如下:
Listresult = new ArrayList<>( orders.stream() .collect(Collectors.toMap( order -> new IdAndDate(order.getId(), order.getDate()), Function.identity(), Order::combine // ⚠️ 修改原对象 )) .values() );
⚠️ 注意事项:可变性与线程安全
当前 Order.combine(Order other) 方法返回 this 并直接修改当前实例(如 setAmount(getAmount() + other.getAmount())),这意味着:
- 原始 orders 列表中的部分对象状态会被改变;
- 若原始数据需保持不变,或在并行流(.parallelStream())中使用,该实现不安全。
✅ 推荐改进:返回新实例(不可变风格)
public Order combine(Order other) {
return new Order(
this.id,
this.date,
this.amount + other.amount
);
}此时需确保 Order 类提供对应构造器(或使用 Builder),并保证 IdAndDate 键的 equals/hashCode 正确(record 已自动实现)。这样既保持函数式编程的纯净性,也支持并行处理。
? 替代方案对比(不推荐)
- ❌ 嵌套 groupingBy(如原代码):逻辑冗余、可读性差、性能略低(两次遍历+多层 Map 构建);
- ❌ 先 groupingBy 再 reduce:虽可行,但需手动处理空值和初始值,代码更 verbose;
- ✅ toMap 是标准库中专为此类“去重+合并”场景设计的最优解。
✅ 总结
| 方案 | 简洁性 | 可读性 | 安全性 | 推荐度 |
|---|---|---|---|---|
| toMap + record 键 + combine | ★★★★★ | ★★★★☆ | ⚠️(需改造成不可变) | ⭐⭐⭐⭐⭐ |
| 嵌套 groupingBy | ★★☆☆☆ | ★★☆☆☆ | ★★★☆☆ | ⭐⭐☆☆☆ |
一句话实践建议:优先使用 Collectors.toMap 配合语义化复合键,让合并逻辑集中、直观、高效;若需数据不可变,请让 combine 返回新对象而非修改自身。
# java
# app
# stream
# 标准库
# red
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
js代码实现下拉菜单【推荐】
WEB开发之注册页面验证码倒计时代码的实现
Laravel如何处理文件下载请求?(Response示例)
Laravel安装步骤详细教程_Laravel环境搭建指南
iOS UIView常见属性方法小结
php在windows下怎么调试_phpwindows环境调试操作说明【操作】
Laravel如何操作JSON类型的数据库字段?(Eloquent示例)
开心动漫网站制作软件下载,十分开心动画为何停播?
Laravel如何发送系统通知_Laravel Notifications实现多渠道消息通知
如何在香港服务器上快速搭建免备案网站?
使用spring连接及操作mongodb3.0实例
阿里云网站搭建费用解析:服务器价格与建站成本优化指南
如何快速搭建二级域名独立网站?
Python制作简易注册登录系统
微信h5制作网站有哪些,免费微信H5页面制作工具?
如何在服务器上三步完成建站并提升流量?
Laravel如何安装Breeze扩展包_Laravel用户注册登录功能快速实现【流程】
html5audio标签播放结束怎么触发事件_onended回调方法【教程】
Laravel如何实现多级无限分类_Laravel递归模型关联与树状数据输出【方法】
Laravel全局作用域是什么_Laravel Eloquent Global Scopes应用指南
Laravel Blade模板引擎语法_Laravel Blade布局继承用法
如何在橙子建站上传落地页?操作指南详解
Windows10如何删除恢复分区_Win10 Diskpart命令强制删除分区
移动端脚本框架Hammer.js
Laravel如何发送系统通知?(Notification渠道示例)
详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)
如何制作一个表白网站视频,关于勇敢表白的小标题?
电商网站制作价格怎么算,网上拍卖流程以及规则?
详解CentOS6.5 安装 MySQL5.1.71的方法
动图在线制作网站有哪些,滑动动图图集怎么做?
详解jQuery中的事件
青岛网站建设如何选择本地服务器?
Laravel如何生成和使用数据填充?(Seeder和Factory示例)
Android自定义控件实现温度旋转按钮效果
Laravel如何生成API文档?(Swagger/OpenAPI教程)
Laravel如何与Pusher实现实时通信?(WebSocket示例)
高端建站三要素:定制模板、企业官网与响应式设计优化
在Oracle关闭情况下如何修改spfile的参数
Laravel如何部署到服务器_线上部署Laravel项目的完整流程与步骤
如何使用 Go 正则表达式精准提取括号内首个纯字母标识符(忽略数字与嵌套)
高端智能建站公司优选:品牌定制与SEO优化一站式服务
HTML5打空格有哪些误区_新手常犯的空格使用错误【技巧】
黑客入侵网站服务器的常见手法有哪些?
Laravel如何实现一对一模型关联?(Eloquent示例)
JS中页面与页面之间超链接跳转中文乱码问题的解决办法
详解阿里云nginx服务器多站点的配置
Laravel怎么创建自己的包(Package)_Laravel扩展包开发入门到发布
常州企业网站制作公司,全国继续教育网怎么登录?
php打包exe后无法访问网络共享_共享权限设置方法【教程】
Laravel用户密码怎么加密_Laravel Hash门面使用教程


)。