[Java8新特性]Collectors源码阅读-3 groupingBy和partitioningBy

本文详细介绍了Java8中Collectors的groupingBy和partitioningBy方法,包括它们的不同重载版本及其实现原理。通过对Function和Predicate的使用,实现数据的分组和分区,并可结合下游Collector进行进一步处理。groupingBy方法通过classifier对数据进行分组,而partitioningBy则依据predicate将数据分为满足条件和不满足条件的两部分。这两个方法在实现中都涉及到Collector的构造,包括supplier、accumulator、combiner和finisher等关键部分。
摘要由CSDN通过智能技术生成

本节介绍两个比较复杂的函数groupingBy, partitioningBy,这两个函数也是通过new CollectorImpl来实现的,分别有多个重载,下面会详细介绍

1.groupingBy

这个方法可以用来分组,类似于sql中的group by,按照指定的键分组,比如:

Map<City, List<Person>> groupByCity
     = people.stream().collect(groupingBy(Person::getCity);

这个操作能将流中的所有people按照city来分组,每个分组是一个List,里面的person对象的city是相同的
这个方法有三个重载方法,前两个比较简单的方法是通过调用第三个方法来实现的,第三个方法才有具体的实现,下面来一一介绍

1.1 groupingBy(Function<? super T, ? extends K> classifier)

这个方法接受一个Function,传入类型T,返回类型K,这个方法叫classifier,就是用来划分组和组之间的分类器

groupingBy返回的类型是Collector<T, ?, Map<K, List>>,这个collector接受的类型是T,返回的类型是Map<K, List>,Map的key类型是K,value类型是List
举个例子,还是刚刚的代码,这个T类型就是City,K就是Person

Map<City, List<Person>> groupByCity
     = people.stream().collect(groupingBy(Person::getCity);

很容易理解,按city分组之后,对于某一组来说,key自然是这个city,value就是这个分组下的Person
这个方法是借助第二个重载方法实现的,详细的在下个小节介绍

public static <T, K> Collector<T, ?, Map<K, List<T>>>
groupingBy(Function<? super T, ? extends K> classifier) {
   
    return groupingBy(classifier, toList());
}
1.2 groupingBy(Function<? super T, ? extends K> classifier,Collector<? super T, A, D> downstream)

这个方法接受两个参数,第一个是classifier,第二个参数是一个Collector,叫做downstream,顾名思义,就是一个下游的Collector,这里我们可以理解为分完组之后对组中的元素进行的后续处理。
还是刚刚的例子,对于上面的那个方法分组后得到Map<City, List<Person>,downStream可以对List<Persion>进行二次处理,比如如果想让people这个list在按City分组之后,取每个person 的lastName,并且希望lastName的容器类型由List变成Set,可以这样做:

Map<City, Set<String>> namesByCity
	= people.stream().collect(groupingBy(Person::getCity, mapping(Person::getLastName, toSet()
  • 3
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Java8中,我们可以使用Stream API来对集合进行分组、排序等操作,其中Collectors.groupingBy是一个非常有用的方法。它接收一个Function作为参数,用于将元素映射为分组的键,返回一个Map类型的结果,其中键是由Function映射的结果,值是对应的元素列表。 例如,我们有一个Person类,其中包含姓名和年龄两个属性,我们想按照年龄进行分组: ``` List<Person> persons = Arrays.asList( new Person("John", 30), new Person("Jane", 25), new Person("Bob", 40), new Person("Mary", 35) ); Map<Integer, List<Person>> personsByAge = persons.stream() .collect(Collectors.groupingBy(Person::getAge)); personsByAge.forEach((age, list) -> System.out.println(age + " -> " + list)); ``` 输出结果为: ``` 25 -> [Person{name='Jane', age=25}] 30 -> [Person{name='John', age=30}] 35 -> [Person{name='Mary', age=35}] 40 -> [Person{name='Bob', age=40}] ``` 我们还可以对分组结果进行排序,例如按照年龄从小到大排序: ``` Map<Integer, List<Person>> personsByAgeSorted = persons.stream() .collect(Collectors.groupingBy(Person::getAge, TreeMap::new, Collectors.toList())); personsByAgeSorted.forEach((age, list) -> System.out.println(age + " -> " + list)); ``` 输出结果为: ``` 25 -> [Person{name='Jane', age=25}] 30 -> [Person{name='John', age=30}] 35 -> [Person{name='Mary', age=35}] 40 -> [Person{name='Bob', age=40}] ``` 在这个例子中,我们使用TreeMap作为分组结果的容器,实现了按照年龄从小到大排序的功能。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值