在路上

 找回密码
 立即注册
在路上 站点首页 学习 查看内容

Java 8 Streams API:对Stream分组和分区

2017-2-7 13:41| 发布者: zhangjf| 查看: 472| 评论: 0

摘要: 这篇文章展示了如何使用 Streams API 中的 Collector 及 groupingBy 和 partitioningBy 来对流中的元素进行分组和分区。 思考一下 Employee 对象流,每个对象对应一个名字、城市和销售数量,如下表所示:+---------- ...

这篇文章展示了如何使用 Streams API 中的 Collector 及 groupingBy 和 partitioningBy 来对流中的元素进行分组和分区。

思考一下 Employee 对象流,每个对象对应一个名字、城市和销售数量,如下表所示:

  1. +----------+------------+-----------------+
  2. | Name | City | Number of Sales |
  3. +----------+------------+-----------------+
  4. | Alice | London | 200 |
  5. | Bob | London | 150 |
  6. | Charles | New York | 160 |
  7. | Dorothy | Hong Kong | 190 |
  8. +----------+------------+-----------------+
复制代码
分组

首先,我们利用(lambda表达式出现之前的)命令式风格Java 程序对流中的雇员按城市进行分组:

  1. Map<String, List<Employee>> result = new HashMap<>();
  2. for (Employee e : employees) {
  3. String city = e.getCity();
  4. List<Employee> empsInCity = result.get(city);
  5. if (empsInCity == null) {
  6. empsInCity = new ArrayList<>();
  7. result.put(city, empsInCity);
  8. }
  9. empsInCity.add(e);
  10. }
复制代码

你可能很熟悉写这样的代码,你也看到了,一个如此简单的任务就需要这么多代码!

而在 Java 8 中,你可以使用 groupingBy 收集器,一条语句就能完成相同的功能,像这样:

  1. Map<String, List<Employee>> employeesByCity =
  2. employees.stream().collect(groupingBy(Employee::getCity));
复制代码

结果如下面的 map 所示:

  1. {New York=[Charles], Hong Kong=[Dorothy], London=[Alice, Bob]}
复制代码

还可以计算每个城市中雇员的数量,只需传递一个计数收集器给 groupingBy 收集器。第二个收集器的作用是在流分类的同一个组中对每个元素进行递归操作。

  1. Map<String, Long> numEmployeesByCity =
  2. employees.stream().collect(groupingBy(Employee::getCity, counting()));
复制代码

结果如下面的 map 所示:

  1. {New York=1, Hong Kong=1, London=2}
复制代码

顺便提一下,该功能与下面的 SQL 语句是等同的:

  1. select city, count(*) from Employee group by city
复制代码

另一个例子是计算每个城市的平均年龄,这可以联合使用 averagingInt 和 groupingBy 收集器:

  1. Map<String, Double> avgSalesByCity =
  2. employees.stream().collect(groupingBy(Employee::getCity,
  3. averagingInt(Employee::getNumSales)));
复制代码

结果如下 map 所示:

  1. {New York=160.0, Hong Kong=190.0, London=175.0}
复制代码
分区

分区是一种特殊的分组,结果 map 至少包含两个不同的分组——一个true,一个false。例如,如果想找出最优秀的员工,你可以将所有雇员分为两组,一组销售量大于 N,另一组小于 N,使用 partitioningBy 收集器:

  1. Map<Boolean, List<Employee>> partitioned =
  2. employees.stream().collect(partitioningBy(e -> e.getNumSales() > 150));
复制代码

输出如下结果:

  1. {false=[Bob], true=[Alice, Charles, Dorothy]}
复制代码

你也可以将 groupingBy 收集器传递给 partitioningBy 收集器来将联合使用分区和分组。例如,你可以统计每个分区中的每个城市的雇员人数:

  1. Map<Boolean, Map<String, Long>> result =
  2. employees.stream().collect(partitioningBy(e -> e.getNumSales() > 150,
  3. groupingBy(Employee::getCity, counting())));
复制代码

这样会生成一个二级 Map:

  1. {false={London=1}, true={New York=1, Hong Kong=1, London=1}}
复制代码

原文链接: javacodegeeks 翻译:ImportNew.com -paddx

译文链接:[]

来自: http://www.importnew.com/17313.html

最新评论

小黑屋|在路上 ( 蜀ICP备15035742号-1 

;

GMT+8, 2025-7-9 09:36

Copyright 2015-2025 djqfx

返回顶部