JavaStream嵌套列表转Map技巧
哈喽!大家好,很高兴又见面了,我是golang学习网的一名作者,今天由我给大家带来一篇《Java Stream分组嵌套列表为Map方法》,本文主要会讲到等等知识点,希望大家一起学习进步,也欢迎大家关注、点赞、收藏、转发! 下面就一起来看看吧!
1. 问题背景与挑战
在处理复杂数据结构时,我们经常遇到需要根据嵌套对象(如列表中的列表)的属性来对外部对象进行分组的需求。例如,给定一个Trip对象列表,每个Trip包含一个Employee对象列表,目标是创建一个Map
直接尝试使用Collectors.groupingBy对Trip流进行分组,并尝试从Trip中获取员工ID列表作为键,通常会导致编译错误或不符合预期的结果。这是因为groupingBy期望一个单一的、可作为键的值,而不是一个流或列表。例如,将t.getEmpList().stream().map(Employee::getEmpId)作为groupingBy的分类函数,会导致键类型为Stream
2. 解决方案核心思路:扁平化与辅助对象
解决此问题的关键在于:
- 扁平化流: 将Stream
转换为一个更细粒度的流,其中每个元素能够直接关联到员工ID和对应的Trip。 - 辅助对象: 引入一个临时的数据结构(如Java 16的record或一个简单的POJO类),用于将每个Employee的empId与其所属的Trip实例进行绑定。
通过这种方式,我们可以将“一个Trip包含多个Employee”的“一对多”关系,转换为“一个TripEmployee实例代表一个员工在一次行程中的参与”,从而使得后续的分组操作变得简单明了。
3. 定义数据模型
首先,我们定义问题中涉及的领域模型:
import lombok.AllArgsConstructor; import lombok.Data; import lombok.NoArgsConstructor; import java.util.Date; import java.util.List; @Data @NoArgsConstructor @AllArgsConstructor public class Trip { private Date startTime; private Date endTime; List<Employee> empList; } @Data @NoArgsConstructor @AllArgsConstructor public class Employee { private String name; private String empId; }
为了辅助分组,我们引入一个record(Java 16+)或一个简单的类来关联员工ID和行程:
// 使用Java 16+ 的 record public record TripEmployee(String empId, Trip trip) {} // 对于Java 8-15,可以使用一个普通的类 /* public class TripEmployee { private String empId; private Trip trip; public TripEmployee(String empId, Trip trip) { this.empId = empId; this.trip = trip; } public String getEmpId() { return empId; } public Trip getTrip() { return trip; } // 可以根据需要添加equals, hashCode, toString } */
record的优势在于其简洁性,编译器会自动生成构造函数、访问器、equals()、hashCode()和toString()方法。
4. 使用Stream API进行数据转换与分组
核心的Stream管道将分为两步:
4.1 步骤一:扁平化流 (flatMap)
我们首先将Stream
trips.stream() .flatMap(trip -> trip.getEmpList().stream() // 将每个Trip的empList转换为Stream<Employee> .map(emp -> new TripEmployee(emp.getEmpId(), trip)) // 将每个Employee映射为TripEmployee ) // 此时流的类型为 Stream<TripEmployee>
flatMap操作在这里至关重要,它将一个Stream
4.2 步骤二:分组聚合 (groupingBy 与 mapping)
在得到Stream
- 分类函数: TripEmployee::empId,这会根据empId进行分组。
- 下游收集器: 由于我们希望每个empId对应一个List
,而当前流中的元素是TripEmployee,我们需要使用Collectors.mapping来提取Trip对象。mapping收集器需要一个映射函数(TripEmployee::trip)和一个最终的下游收集器(Collectors.toList())来将提取出的Trip收集成列表。
.collect(Collectors.groupingBy( TripEmployee::empId, // 根据empId进行分组 Collectors.mapping(TripEmployee::trip, // 将TripEmployee映射为Trip Collectors.toList()) // 将映射后的Trip收集为List ));
5. 完整示例代码
以下是包含数据初始化和完整Stream管道的示例:
import lombok.AllArgsConstructor; import lombok.Data; import lombok.NoArgsConstructor; import java.util.ArrayList; import java.util.Date; import java.util.List; import java.util.Map; import java.util.stream.Collectors; // 假设 Trip 和 Employee 类已定义如上 // 辅助记录 (Java 16+) public record TripEmployee(String empId, Trip trip) {} public class TripGroupingExample { public static void main(String[] args) { // 示例数据 Employee emp1 = new Employee("Alice", "E001"); Employee emp2 = new Employee("Bob", "E002"); Employee emp3 = new Employee("Charlie", "E003"); Trip trip1 = new Trip(new Date(), new Date(), List.of(emp1, emp2)); Trip trip2 = new Trip(new Date(), new Date(), List.of(emp1, emp3)); Trip trip3 = new Trip(new Date(), new Date(), List.of(emp2)); Trip trip4 = new Trip(new Date(), new Date(), List.of(emp3, emp1)); // 再次包含emp1 List<Trip> trips = new ArrayList<>(); trips.add(trip1); trips.add(trip2); trips.add(trip3); trips.add(trip4); // 使用Stream API生成Map<String, List<Trip>> Map<String, List<Trip>> empTripsMap = trips.stream() .flatMap(trip -> trip.getEmpList().stream() // 将每个Trip的empList扁平化为Stream<Employee> .map(emp -> new TripEmployee(emp.getEmpId(), trip)) // 将每个Employee映射为TripEmployee ) .collect(Collectors.groupingBy( TripEmployee::empId, // 根据TripEmployee的empId进行分组 Collectors.mapping(TripEmployee::trip, // 将TripEmployee映射回Trip Collectors.toList()) // 将映射后的Trip收集为List )); // 打印结果 empTripsMap.forEach((empId, tripList) -> { System.out.println("Employee ID: " + empId); tripList.forEach(trip -> System.out.println(" - Trip: " + trip)); System.out.println("---"); }); /* 预期输出示例 (具体Trip对象内容取决于toString实现和日期) Employee ID: E001 - Trip: Trip(startTime=..., endTime=..., empList=...) // trip1 - Trip: Trip(startTime=..., endTime=..., empList=...) // trip2 - Trip: Trip(startTime=..., endTime=..., empList=...) // trip4 --- Employee ID: E002 - Trip: Trip(startTime=..., endTime=..., empList=...) // trip1 - Trip: Trip(startTime=..., endTime=..., empList=...) // trip3 --- Employee ID: E003 - Trip: Trip(startTime=..., endTime=..., empList=...) // trip2 - Trip: Trip(startTime=..., endTime=..., empList=...) // trip4 --- */ } }
6. 注意事项与总结
- Java版本兼容性: 示例中使用了Java 16的record,如果您的项目使用Java 8到Java 15,请使用普通的Java类作为辅助对象(如代码注释中所示)。功能上没有区别,只是record提供了更简洁的语法。
- flatMap的重要性: flatMap是处理“一对多”转换的关键操作。它将流中的每个元素映射到一个新的流,然后将这些新的流连接(扁平化)成一个单一的流。在本例中,它将每个Trip转换为多个TripEmployee实例的流,然后合并这些流。
- groupingBy与mapping组合: 当需要根据一个属性分组,但最终值是原始对象或其转换形式的列表时,Collectors.groupingBy结合Collectors.mapping是一个非常强大的模式。mapping允许你在分组之后,对每个组内的元素进行进一步的转换和收集。
- 可读性: 引入TripEmployee这样的辅助对象,虽然增加了一个小类,但显著提高了Stream管道的可读性和意图清晰度,避免了使用Map.Entry等通用但语义不明确的结构。
通过上述方法,我们能够高效且清晰地利用Java Stream API解决从嵌套列表中提取数据并进行复杂分组的问题,使得代码更具表达力和维护性。
好了,本文到此结束,带大家了解了《JavaStream嵌套列表转Map技巧》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多文章知识!

- 上一篇
- Vue.js制作待办事项应用教程

- 下一篇
- some和every都表示“一些”或“每一个”,但用法不同:some:用于肯定句,表示“一些”,不特指哪一个。例句:Ihavesomequestions.(我有一些问题。)every:表示“每一个”,强调全体中的每一个个体。例句:Everystudentpassedtheexam.(每个学生都通过了考试。)总结:some表示“一些”,用于肯定句。every表示“每一个”,强调全部。
-
- 文章 · java教程 | 13分钟前 |
- Java管道流:PipedInputStream与PipedOutputStream详解
- 106浏览 收藏
-
- 文章 · java教程 | 25分钟前 |
- SM4算法Java实现详解教程
- 368浏览 收藏
-
- 文章 · java教程 | 8小时前 | 排序 性能优化 线程安全 Collections工具类 不可变集合
- Java集合工具类使用教程
- 214浏览 收藏
-
- 文章 · java教程 | 9小时前 |
- Java注解处理器代码生成实例解析
- 136浏览 收藏
-
- 文章 · java教程 | 9小时前 | java 配置文件 加载 Properties .properties
- JavaProperties配置加载教程
- 470浏览 收藏
-
- 文章 · java教程 | 9小时前 |
- Java分布式限流算法全解析
- 145浏览 收藏
-
- 文章 · java教程 | 10小时前 |
- Java泛型擦除与通配符详解
- 375浏览 收藏
-
- 文章 · java教程 | 10小时前 |
- Java生成高级Excel报表教程
- 206浏览 收藏
-
- 文章 · java教程 | 11小时前 |
- Selenium关闭广告窗口技巧分享
- 302浏览 收藏
-
- 文章 · java教程 | 11小时前 |
- Java分片上传MinIO实战教程
- 258浏览 收藏
-
- 文章 · java教程 | 11小时前 |
- Java中void方法如何修改布尔值
- 161浏览 收藏
-
- 文章 · java教程 | 11小时前 |
- Java集合操作技巧与使用方法
- 188浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 511次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 498次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 千音漫语
- 千音漫语,北京熠声科技倾力打造的智能声音创作助手,提供AI配音、音视频翻译、语音识别、声音克隆等强大功能,助力有声书制作、视频创作、教育培训等领域,官网:https://qianyin123.com
- 218次使用
-
- MiniWork
- MiniWork是一款智能高效的AI工具平台,专为提升工作与学习效率而设计。整合文本处理、图像生成、营销策划及运营管理等多元AI工具,提供精准智能解决方案,让复杂工作简单高效。
- 218次使用
-
- NoCode
- NoCode (nocode.cn)是领先的无代码开发平台,通过拖放、AI对话等简单操作,助您快速创建各类应用、网站与管理系统。无需编程知识,轻松实现个人生活、商业经营、企业管理多场景需求,大幅降低开发门槛,高效低成本。
- 216次使用
-
- 达医智影
- 达医智影,阿里巴巴达摩院医疗AI创新力作。全球率先利用平扫CT实现“一扫多筛”,仅一次CT扫描即可高效识别多种癌症、急症及慢病,为疾病早期发现提供智能、精准的AI影像早筛解决方案。
- 220次使用
-
- 智慧芽Eureka
- 智慧芽Eureka,专为技术创新打造的AI Agent平台。深度理解专利、研发、生物医药、材料、科创等复杂场景,通过专家级AI Agent精准执行任务,智能化工作流解放70%生产力,让您专注核心创新。
- 241次使用
-
- 提升Java功能开发效率的有力工具:微服务架构
- 2023-10-06 501浏览
-
- 掌握Java海康SDK二次开发的必备技巧
- 2023-10-01 501浏览
-
- 如何使用java实现桶排序算法
- 2023-10-03 501浏览
-
- Java开发实战经验:如何优化开发逻辑
- 2023-10-31 501浏览
-
- 如何使用Java中的Math.max()方法比较两个数的大小?
- 2023-11-18 501浏览