Java8 stream 中利用 groupingBy 进行多字段分组

从简单入手

Stream 作为 Java 8 的一大亮点,好比一个高级的迭代器(Iterator),单向,不可往复,数据只能遍历一次,遍历过一次后即用尽了,就好比流水从面前流过,一去不复返。

我们可以利用stream对数据进行分组。示例如下:

List<String> items = Arrays.asList("apple", "apple", "banana", "apple", "orange", "banana", "papaya");

Map<String, Long> result = items.stream().collect(Collectors.groupingBy(Function.identity(), Collectors.counting());
System.out.println(result);

输出如下:
{papaya=1, orange=1, banana=2, apple=3}

进阶需求

在实际需求中,我们可能需要对一组对象进行分组,而且分组的条件有多个。比如对国家和产品类型进行双重分组,一种比较复杂的方法是先对产品类型进行分组,然后对每一个产品类型中的国家名进行分组求和。示例如下:

Map<String, List<item>> countMap = recordItems.stream().collect(Collectors.groupingBy(o -> o.getProductType()));

List<Record> records = new ArrayList<Record>;
countMap.keySet().forEach(productType -> {
    Map<String, Long> countMap1 = countMap.get(productType).stream().collect(Collectors.groupingBy(o -> o.getCountry(), Collectors.counting()));
    countMap1(key).stream().forEach(country -> {
        Record record = new Record();
        record.set("device_type", productType);
        record.set("location", country;
        record.set("count", countMap1(key).intValue());
        records.add(record);
    });
});

更好的解决方法

上面的方法在应对两个字段的分组要求时,还能应付的过来,但如果字段超过两个时,每增加一个字段,就会多出很多代码行,显然不太合理。更合理的方法是,增加一个 getKey()方法,返回多个字段组成的唯一key,比如通过下划线连接等等。示例如下:


// 分组统计
Map<String, Long> countMap = records.stream().collect(Collectors.groupingBy(o -> o.getProductType() + "_" + o.getCountry(), Collectors.counting()));

List<Record> countRecords = countMap.keySet().stream().map(key -> {
    String[] temp = key.split("_");
    String productType = temp[0];
    String country = temp[1];
    
    Record record = new Record();
    record.set("device_type", productType);
    record.set("location", country;
    record.set("count", countMap.get(key).intValue());
    return record;
}).collect(Collectors.toList());

转载于:https://www.jianshu.com/p/dd5121c8fa89

  • 9
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Java 8引入了Stream API,有许多新方法,其有一个对于分组和聚合操作非常有用,那就是groupingBy()方法。它可以将一个流分组成一个Map,其Entry的key是分组的条件,value是分组的结果,通常是一个List或其他集合。groupingBy()方法的另一个形式是groupingByConcurrent(),它返回一个并发Map,对于并发访问更加友好。 利用groupingBy()方法进行字段分组和求和操作的示例如下: 假设有一个Person类,其包含属性:name, age和salary。现在我们需要根据name和age两个字段进行分组,并求出每组的salary总和。可以使用groupingBy()方法加上summingDouble()方法来实现: ``` List<Person> persons = Arrays.asList( new Person("Tom", 20, 5000), new Person("Tom", 21, 4000), new Person("Jerry", 22, 6000), new Person("Jerry", 23, 5500), new Person("Kate", 24, 7000), new Person("Kate", 25, 8000) ); Map<String, Map<Integer, Double>> result = persons.stream() .collect(Collectors.groupingBy(Person::getName, Collectors.groupingBy(Person::getAge, Collectors.summingDouble(Person::getSalary)))); ``` 这里的personList是一个包含了6个Person对象的List,我们希望将其相同name和age的对象分组,求得salary的总和。在groupingBy()方法,第一个参数是分组条件,这里是Person::getName,第二个参数是分组的结果,这里是一个嵌套的groupingBy()方法,用于再次按照age进行分组,结果是一个Map<Integer, Double>。最后,我们使用summingDouble()方法对salary字段进行求和,得到各个分组的salary总和。这里的result是一个Map<String, Map<Integer, Double>>类型的对象,其key是name,value是以age为key,salary总和为value的子Map,就是我们需要的结果。 这样,我们就利用Java 8Stream API和groupingBy()方法进行了多字段分组和求和操作,代码简洁,可读性强,非常方便。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值