首页 > Java > java教程 > 正文

Java里Stream分组功能怎么实现_Stream在Java中的分组应用说明

P粉602998670
发布: 2025-12-13 02:17:28
原创
411人浏览过
Java中Stream分组主要用Collectors.groupingBy(),返回Map;支持基础分组、二级分组、自定义逻辑(含空值处理)及聚合统计(如counting、averagingDouble),需注意keyMapper类型、null安全与下游收集器选择。

java里stream分组功能怎么实现_stream在java中的分组应用说明

Java中Stream的分组功能主要通过 Collectors.groupingBy() 实现,它能把流中的元素按指定规则分类,返回一个 Map,键是分组依据,值是该组内所有元素的集合。

基础分组:按字段或简单条件分组

最常见的是按对象的某个属性分组。比如有一个 List,想按年龄分组:

Map<Integer, List<Person>> byAge = people.stream()
    .collect(Collectors.groupingBy(Person::getAge));
登录后复制

这时 Map 的 key 是 Integer(年龄),value 是对应年龄的所有 Person 对象组成的 List。

  • 如果属性值为 null,会抛出 NullPointerException,可先用 Objects::nonNull 过滤或用 groupingBy(keyMapper, HashMap::new, …) 自定义 map 类型
  • keyMapper 函数必须返回可比较、可哈希的类型,否则可能出错或分组异常

二级分组:嵌套 groupingBy 实现多级分类

需要按多个维度分组时,可以把另一个 groupingBy 作为下游收集器:

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

Map<Integer, Map<String, List<Person>>> byAgeAndCity = people.stream()
    .collect(Collectors.groupingBy(
        Person::getAge,
        Collectors.groupingBy(Person::getCity)
    ));
登录后复制

结果是 Map>>,适合做交叉统计或层级报表。

  • 下游收集器不限于 groupingBy,也可以是 counting()、summingInt() 等,实现聚合统计
  • 注意嵌套过深会影响可读性,必要时可封装成独立方法

自定义分组逻辑与空值处理

当分组逻辑较复杂(如按年龄段分组:0-18、19-35、36+),或源数据含 null 字段时,推荐用 lambda 显式定义 key:

Map<String, List<Person>> byAgeRange = people.stream()
    .collect(Collectors.groupingBy(p -> {
        int age = p.getAge() == null ? -1 : p.getAge();
        if (age < 0) return "未知";
        else if (age <= 18) return "未成年";
        else if (age <= 35) return "青年";
        else return "成年";
    }));
登录后复制

这样既避免空指针,又灵活支持业务语义分组。

  • 可配合 Collectors.toMap() 或 Collectors.collectingAndThen 做后处理(如排序、去重)
  • 若需保持插入顺序,用 LinkedHashMap::new 作为 map 工厂参数

分组后聚合统计(不存全量数据)

有时只需统计数量、平均值等,不必保留原始列表,节省内存:

// 按部门统计人数
Map<String, Long> countByDept = employees.stream()
    .collect(Collectors.groupingBy(Employee::getDept, Collectors.counting()));

// 按状态统计平均薪资
Map<Status, Double> avgSalaryByStatus = employees.stream()
    .collect(Collectors.groupingBy(
        Employee::getStatus,
        Collectors.averagingDouble(Employee::getSalary)
    ));
登录后复制

这类组合使用让 Stream 分组兼具灵活性和性能优势。

  • counting() 返回 Long,summingInt/Double/Long 返回对应数值类型
  • 若某组无数据,对应 key 仍存在,value 为 0 或 0.0(取决于聚合器)

基本上就这些。Stream 分组不复杂但容易忽略细节,关键是选对 keyMapper 和下游收集器,再结合业务场景做空值和异常处理。

以上就是Java里Stream分组功能怎么实现_Stream在Java中的分组应用说明的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

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