答案:Collectors.groupingBy用于按条件分组,支持多级分组、聚合统计和自定义逻辑。例如按部门分组员工,再嵌套按职位细分,或结合counting、summing等进行数据聚合,也可通过Lambda实现年龄范围等复杂条件分组。

在Java 8及以上版本中,Collectors.groupingBy 是一个非常实用的工具,用于对集合中的元素进行分组。它通常与 Stream API 配合使用,能够将数据按照指定条件分类,生成一个 Map 结构的结果。下面详细介绍其用法和常见场景。
基本语法与简单分组
groupingBy 最简单的形式是根据某个属性或条件对对象进行分组。例如,有一个员工列表,想按部门分组:
Map> employeesByDept = employees.stream() .collect(Collectors.groupingBy(Employee::getDepartment));
这会返回一个以部门名称为键、该部门所有员工列表为值的 Map。每个键对应一组匹配的对象列表。
二级分组:按多个条件分组
如果需要进一步细分,比如先按部门再按职位分组,可以使用 groupingBy 嵌套:
立即学习“Java免费学习笔记(深入)”;
Map>> result = employees.stream() .collect(Collectors.groupingBy( Employee::getDepartment, Collectors.groupingBy(Employee::getPosition) ));
此时结果是一个嵌套 Map:外层 Key 是部门,内层 Key 是职位,最终值是满足两个条件的员工列表。
分组后聚合统计
除了收集元素列表,还可以结合其他收集器做聚合操作。常见的如计数、求和、平均值等:
- 统计每组数量: Collectors.groupingBy(Function.identity(), Collectors.counting())
- 对数值字段求和: Collectors.groupingBy(Employee::getDepartment, Collectors.summingDouble(Employee::getSalary))
- 获取每组最高薪资: Collectors.groupingBy(Employee::getDepartment, Collectors.mapping(Employee::getSalary, Collectors.maxBy(Double::compare)))
自定义分组条件
分组条件不局限于 getter 方法,也可以是任意函数逻辑。例如,将年龄按年龄段分组(如20-29、30-39):
Map> groupedByAgeRange = people.stream() .collect(Collectors.groupingBy(person -> { int age = person.getAge(); if (age < 20) return "Under 20"; else if (age < 30) return "20s"; else if (age < 40) return "30s"; else return "40+"; }));
这种灵活的方式适用于复杂业务规则下的数据归类。
基本上就这些。合理使用 groupingBy 能让集合处理更清晰高效,尤其适合报表生成、数据分析等场景。关键是理解其函数式结构,并搭配合适的下游收集器完成聚合目标。










