0

0

Java 8 Stream 实现按 ID 和日期分组并合并同组订单金额

碧海醫心

碧海醫心

发布时间:2026-01-01 12:20:40

|

240人浏览过

|

来源于php中文网

原创

Java 8 Stream 实现按 ID 和日期分组并合并同组订单金额

本文介绍如何使用 java 8 stream 的 `collectors.tomap` 配合自定义键(如 `record idanddate`)与合并函数,高效地对订单列表按 `id` 和 `date` 两字段分组,并原地或不可变地聚合 `amount`,最终直接获得合并后的 `order` 列表,避免嵌套 `groupingby` 和中间 `map>` 结构。

在 Java 8 Stream 中,若需按多个字段联合分组(如 id + date)并合并同组元素(如累加 amount),最简洁、高效的方式并非嵌套 groupingBy,而是使用 Collectors.toMap —— 它天然支持键冲突时的自定义合并逻辑(即 mergeFunction),且可直接产出 Map,再通过 .values() 转为所需 List

✅ 推荐方案: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 合并金额(注意:此方法当前为就地修改)。

完整代码如下:

立即学习Java免费学习笔记(深入)”;

微软爱写作
微软爱写作

微软出品的免费英文写作/辅助/批改/评分工具

下载
List<Order> result = 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 返回新对象而非修改自身。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

764

2023.08.10

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

77

2025.09.05

golang map相关教程
golang map相关教程

本专题整合了golang map相关教程,阅读专题下面的文章了解更多详细内容。

40

2025.11.16

golang map原理
golang map原理

本专题整合了golang map相关内容,阅读专题下面的文章了解更多详细内容。

67

2025.11.17

java判断map相关教程
java判断map相关教程

本专题整合了java判断map相关教程,阅读专题下面的文章了解更多详细内容。

47

2025.11.27

function是什么
function是什么

function是函数的意思,是一段具有特定功能的可重复使用的代码块,是程序的基本组成单元之一,可以接受输入参数,执行特定的操作,并返回结果。本专题为大家提供function是什么的相关的文章、下载、课程内容,供大家免费下载体验。

498

2023.08.04

js函数function用法
js函数function用法

js函数function用法有:1、声明函数;2、调用函数;3、函数参数;4、函数返回值;5、匿名函数;6、函数作为参数;7、函数作用域;8、递归函数。本专题提供js函数function用法的相关文章内容,大家可以免费阅读。

166

2023.10.07

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

4

2026.03.10

Kotlin Android模块化架构与组件化开发实践
Kotlin Android模块化架构与组件化开发实践

本专题围绕 Kotlin 在 Android 应用开发中的架构实践展开,重点讲解模块化设计与组件化开发的实现思路。内容包括项目模块拆分策略、公共组件封装、依赖管理优化、路由通信机制以及大型项目的工程化管理方法。通过真实项目案例分析,帮助开发者构建结构清晰、易扩展且维护成本低的 Android 应用架构体系,提升团队协作效率与项目迭代速度。

25

2026.03.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11万人学习

Java 教程
Java 教程

共578课时 | 79.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号