java8中使用groupingBy分组返回有序的Map

分组后返回有序的Map:
使用 (Collectors.groupingBy(User::getType , LinkedHashMap::new, Collectors.toList()))

LinkedHashMap取键值对时,是按照放入的顺序

LinkedHashMap<String, List<User>> groupMap2 = list.stream().collect(Collectors.groupingBy(User::getType, LinkedHashMap::new, Collectors.toList()));

groupby方法有几个重载方法,上面使用的这个方法有3个参数

  1. 第一个参数:分组按照什么进行分类
  2. 第二个参数:分组结果最后用什么容器保存并返回,这里指定为LinkedHashMap
  3. 第三个参数:分类后,对应的分类的结果如何收集
package com.test;

public class User {
    private Integer id;
    private String name;
    private String type;

    public User() {
    }

    public User(Integer id, String name, String type) {
        this.id = id;
        this.name = name;
        this.type = type;
    }

    public Integer getId() {
        return id;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getType() {
        return type;
    }

    public void setType(String type) {
        this.type = type;
    }

    @Override
    public String toString() {
        return "User{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", type='" + type + '\'' +
                '}';
    }
}
package com.test;

import java.util.*;
import java.util.stream.Collectors;

public class MainUser {
    /**
     * 构造一个集合
     */
    public static List<User> getUserList() {
        User user1 = new User(1, "张", "A");
        User user2 = new User(2, "李", "BB");
        User user3 = new User(3, "王", "BB");
        User user4 = new User(4, "马", "CCC");
        User user5 = new User(5, "赵", "CCC");
        User user6 = new User(6, "钱", "CCC");

        List<User> list = new ArrayList<User>();
        list.add(user2);
        list.add(user4);
        list.add(user3);
        list.add(user1);
        list.add(user5);
        list.add(user6);

        return list;
    }

    /**
     * 打印出所有的 key 及 对应list中的元素
     */
    private static void print(Map<String, List<User>> groupMap) {
        groupMap.forEach((key, list) -> {
            System.out.println(key);
            list.forEach(System.out::println);
        });
    }

    public static void main(String[] args) {
        List<User> list = getUserList();
        //根据 id 升序
        list = list.stream().sorted(Comparator.comparing(User::getId)).collect(Collectors.toList());
        list.forEach(System.out::println);
        System.out.println();

        //(1)这样分组,key 是无序的
        Map<String, List<User>> groupMap1 = list.stream().collect(Collectors.groupingBy(User::getType));
        print(groupMap1);

        System.out.println();

        //(2)这样分组,key 还是原来的顺序
        Map<String, List<User>> groupMap2 = list.stream().collect(Collectors.groupingBy(User::getType, LinkedHashMap::new, Collectors.toList()));
// LinkedHashMap<String, List<User>> groupMap2 = list.stream().collect(Collectors.groupingBy(User::getType, LinkedHashMap::new, Collectors.toList()));
        print(groupMap2);
    }
}

输出结果:

User{id=1, name='张', type='A'}
User{id=2, name='李', type='BB'}
User{id=3, name='王', type='BB'}
User{id=4, name='马', type='CCC'}
User{id=5, name='赵', type='CCC'}
User{id=6, name='钱', type='CCC'}

BB
User{id=2, name='李', type='BB'}
User{id=3, name='王', type='BB'}
A
User{id=1, name='张', type='A'}
CCC
User{id=4, name='马', type='CCC'}
User{id=5, name='赵', type='CCC'}
User{id=6, name='钱', type='CCC'}

A
User{id=1, name='张', type='A'}
BB
User{id=2, name='李', type='BB'}
User{id=3, name='王', type='BB'}
CCC
User{id=4, name='马', type='CCC'}
User{id=5, name='赵', type='CCC'}
User{id=6, name='钱', type='CCC'}

主要就是
(Collectors.groupingBy(User::getType))
修改为
(Collectors.groupingBy(User::getType , LinkedHashMap::new, Collectors.toList()))

  • 3
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
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 8的Stream API和groupingBy()方法进行了多字段分组和求和操作,代码简洁,可读性强,非常方便。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值