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

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

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

本节介绍两个比较复杂的函数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()
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值