Java 8 lambda Collectors优雅的使用方式-超详细-groupingBy分组自定义比较器

背景

在使用Java8进行分组的时候,有的时候需要分组返回的数据是固定的,这个就需要指定分组使用的map类型,以下是一些案例

数据准备 

实体类 (Student)


public class Student {
        /**
         * 班级
         */
        private String grade;
        /**
         * 年龄
         */
        private int age;

        public Student(String grade, int age) {
            this.grade = grade;
            this.age = age;
        }

        public String getGrade() {
            return grade;
        }

        public void setGrade(String grade) {
            this.grade = grade;
        }

        public int getAge() {
            return age;
        }

        public void setAge(int age) {
            this.age = age;
        }
    }

排序枚举类 

public enum SortEnum {
        SIX("六年级", 1),
        SEVEN("七年级", 2),
        NINE("九年级", 3),
        OTHER("其他", -1);

        private SortEnum(String name, Integer value) {
            this.value = value;
            this.name = name;
        }

        public Integer getValue() {
            return value;
        }

        private Integer value;

        public String getName() {
            return name;
        }

        private String name;

        public static Integer getSort(String name) {
            if (name == null) {
                return OTHER.getValue();
            }
            SortEnum[] instances = SortEnum.values();
            for (SortEnum i : instances) {
                if (name != null && name.equals(i.getName())) {
                    return i.getValue();
                }
            }
            return OTHER.getValue();
        }
    }

 

测试类 (Main方法)

    public static void main(String[] args) {
        Student t1 = new Student("六年级", 23);
        Student t2 = new Student("六年级", 14);
        Student t3 = new Student("六年级", 56);
        Student t4 = new Student("九年级", 56);
        Student t5 = new Student("九年级", 16);
        Student t6 = new Student("九年级", 36);
        Student t7 = new Student("七年级", 56);
        Student t8 = new Student("七年级", 16);
        Student t9 = new Student("七年级", 36);
        List<Student> list = Lists.newArrayList(t1, t2, t3, t4, t5, t6, t7, t8, t9);

        //根据班级分组
        list.stream().collect(Collectors.groupingBy(Student::getGrade));
}

分组后的集合是有序的写法

根据集合中元素的年龄属性排序

Map<String, List<Student>> collect1 = list.stream().sorted(Comparator.comparingInt(Student::getAge))
.collect(Collectors.groupingBy(Student::getGrade));
System.out.println(collect1);
//输出
{九年级=[Student{grade='九年级', age=16}, Student{grade='九年级', age=36}, Student{grade='九年级', age=56}],
七年级=[Student{grade='七年级', age=16}, Student{grade='七年级', age=36}, Student{grade='七年级', age=56}],
六年级=[Student{grade='六年级', age=14}, Student{grade='六年级', age=23}, Student{grade='六年级', age=56}]}

需要分组的key是有序的

分组后发现年级是无序的 ,Collectors中的groupingBy有三个重载方法,默认分组后的数据是HashMap导致显示的时候是无序的,分组的集合默认是List,所以如果想要key是有序的话,就需要使用有序的Map,可以选择LinkedHashMap或者是TreeMap

 

 LinkedHashMap写法

按照放进去的先后显示能满足数据不会变化的要求,但是不能满足排序的要求

Map<String, List<Student>> collect2 = list.stream().collect(
Collectors.groupingBy(Student::getGrade, LinkedHashMap::new, Collectors.toList()));

TreeMap写法

能满足排序的要求,默认按照自然排序的方式显示,如果是复杂对象就需要传入一个比较器进行比较

Map<String, List<Student>> collect3 = list.stream().collect(
Collectors.groupingBy(Student::getGrade, TreeMap::new, Collectors.toList()));

 TreeMap写法(自定义比较器)

Map<String, List<Student>> collect4 = list.stream().collect(Collectors.groupingBy(
Student::getGrade,() -> new TreeMap<>(Comparator.comparing(SortEnum::getSort)), Collectors.toList()));

 来看一下这个写法的演变过程

其实就是将其中的代码替换成对应的lambda表达式

Map<String, List<Student>> collect4 = list.stream().sorted(Comparator.comparingInt(Student::getAge)).collect(
Collectors.groupingBy(Student::getGrade, () -> {
                    return new TreeMap<>(new Comparator<String>() {
                        @Override
                        public int compare(String o1, String o2) {
                            return SortEnum.getSort(o1).compareTo(SortEnum.getSort(o2));
                        }
                    });
                }, Collectors.toList()));

 

  Map<String, List<Student>> collect4 = list.stream().sorted(Comparator.comparingInt(Student::getAge)).collect(
Collectors.groupingBy(Student::getGrade, () -> {
                    return new TreeMap<>((o1, o2) -> SortEnum.getSort(o1).compareTo(SortEnum.getSort(o2)));
                }, Collectors.toList()));
Map<String, List<Student>> collect4 = list.stream().sorted(Comparator.comparingInt(Student::getAge)).collect(
Collectors.groupingBy(Student::getGrade, () -> new TreeMap<>((o1, o2) ->
 SortEnum.getSort(o1).compareTo(SortEnum.getSort(o2))), Collectors.toList()));
 Map<String, List<Student>> collect4 = list.stream().sorted(Comparator.comparingInt(Student::getAge)).collect(
Collectors.groupingBy(Student::getGrade, () -> new TreeMap<>(Comparator.comparing(SortEnum::getSort)), Collectors.toList()));

因为枚举中设置了序号,因此输出的结果是

//LinkedMap输出 - 集合未排序
{六年级=[Student{grade='六年级', age=23}, Student{grade='六年级', age=14}, Student{grade='六年级', age=56}], 
九年级=[Student{grade='九年级', age=56}, Student{grade='九年级', age=16}, Student{grade='九年级', age=36}], 
七年级=[Student{grade='七年级', age=56}, Student{grade='七年级', age=16}, Student{grade='七年级', age=36}]}

//TreeMapMap默认输出- 集合未排序
{七年级=[Student{grade='七年级', age=56}, Student{grade='七年级', age=16}, Student{grade='七年级', age=36}], 
九年级=[Student{grade='九年级', age=56}, Student{grade='九年级', age=16}, Student{grade='九年级', age=36}], 
六年级=[Student{grade='六年级', age=23}, Student{grade='六年级', age=14}, Student{grade='六年级', age=56}]}

//TreeMap自定义比较器输出 - 集合已排序
{六年级=[Student{grade='六年级', age=14}, Student{grade='六年级', age=23}, Student{grade='六年级', age=56}], 
七年级=[Student{grade='七年级', age=16}, Student{grade='七年级', age=36}, Student{grade='七年级', age=56}], 
九年级=[Student{grade='九年级', age=16}, Student{grade='九年级', age=36}, Student{grade='九年级', age=56}]}

 

  • 1
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值