使用收集器
转换成其他集合
IntStream.range(0,5).boxed().collect(Collectors.toCollection(TreeSet::new));
转换成值
IntStream.range(0,5).boxed().collect(Collectors.maxBy(Comparator.comparing(x->x)))
数据分块
![](https://i-blog.csdnimg.cn/blog_migrate/4dc46a978a215804bc45242a7065b490.png)
数据分组
![](https://i-blog.csdnimg.cn/blog_migrate/02fefbcc2fc12acafcb4254cbc07ea2c.png)
工厂方法 | 返回类型 | 用于 |
toList | List<T> | 把流中所有项目收集到一个List |
使用示例: List<Dish> dishes = menuStream.collect(toList()); | ||
toSet | Set<T> | 把流中所有项目收集到一个 Set,删除重复项 |
使用示例: Set<Dish> dishes = menuStream.collect(toSet()); | ||
toCollection | Collection<T> | 把流中所有项目收集到给定的供应源创建的集合 |
使用示例: Collection<Dish> dishes = menuStream.collect(toCollection(),ArrayList::new); | ||
counting | Long | 计算流中元素的个数 |
使用示例: long howManyDishes = menuStream.collect(counting()); | ||
summingInt | Integer | 对流中项目的一个整数属性求和 |
使用示例: int totalCalories = menuStream.collect(summingInt(Dish::getCalories)) | ||
averagingInt | Double | 计算流中项目 Integer 属性的平均值 |
使用示例: double avgCalories = menuStream.collect(averagingInt(Dish::getCalories)); | ||
summarizingInt | IntSummaryStatistics | 收集关于流中项目 Integer 属性的统计值,例如最大、最小、总和与平均值 |
使用示例 : IntSummaryStatistics menuStatistics=menuStream.collect(summarizingInt(Dish::getCalories)); | ||
joining | String | 连接对流中每个项目调用 toString 方法所生成的字符串 |
使用示例: String shortMenu =menuStream.map(Dish::getName).collect(joining(", ")); | ||
maxBy | Optional<T> | 一个包裹了流中按照给定比较器选出的最大元素的 Optional,或如果流为空则为 Optional.empty() |
使用示例: Optional<Dish> fattest =menuStream.collect(maxBy(comparingInt(Dish::getCalories))); | ||
minBy | Optional<T> | 一个包裹了流中按照给定比较器选出的最小元素的 Optional,或如果流为空则为 Optional.empty() |
使用示例: Optional<Dish> lightest =menuStream.collect(minBy(comparingInt(Dish::getCalories))); | ||
reducing | 归约操作产生的类型 | 从一个作为累加器的初始值开始,利用 BinaryOperator 与流中的元素逐个结合,从而将流归约为单个值 |
使用示例: int totalCalories =menuStream.collect(reducing(0, Dish::getCalories, Integer::sum)); | ||
collectingAndThen | 转换函数返回的类型 | 包裹另一个收集器,对其结果应用转换函数 |
使用示例: int howManyDishes =menuStream.collect(collectingAndThen(toList(), List::size) | ||
groupingBy | Map<K, List<T>> | 根据项目的一个属性的值对流中的项目作问组,并将属性值作为结果 Map 的键 |
使用示例: Map<Dish.Type,List<Dish>>dishesByType =menuStream.collect(groupingBy(Dish::getType)); | ||
partitioningBy | Map<Boolean,List<T>> | 根据对流中每个项目应用谓词的结果来对项目进行分区 |
使用示例: Map<Boolean,List<Dish>> vegetarianDishes =menuStream.collect(partitioningBy(Dish::isVegetarian)); |
组合收集器
各种收集器已经很强大,但是仍然可以将它们组合起来使用,如下面这个例子:
public Map<Artist,Long> numberOfAlbums(Stream<Album> albums){
return albums.collect(groupingBy(album->album.getMainMusician(),counting()));
}
public Map<Artist,List<String>> numberOfAlbums(Stream<Album> albums){
return albums.collect(groupingBy(album->album.getMainMusician(),mapping(Album::getName,toList())));
}
收集器接口
public interface Collector<T, A, R> {
Supplier<A> supplier();
BiConsumer<A, T> accumulator();
BinaryOperator<A> combiner();
Function<A, R> finisher();
Set<java.util.stream.Collector.Characteristics> characteristics();
}
public class ToListCollector<T> implements Collector<T, List<T>, List<T>>
supplier方法必须返回一个结果为空的Supplier,也就是一个无参数函数,在调用时它会创建一个空的累加器实例,供数据收集过程使用。很明显,对于将累加器本身作为结果返回的收集器,比如我们的ToListCollector,在对空流执行操作的时候,这个空的累加器也代表了收集过程的结果。在我们的ToListCollector中,supplier返回一个空的List,如下所示:
public Supplier<List<T>> supplier() {
return () -> new ArrayList<T>();
}
accumulator方法会返回执行归约操作的函数。当遍历到流中第n个元素时,这个函数执行时会有两个参数:保存归约结果的累加器(已收集了流中的前 n-1 个项目), 还有第n个元素本身。该函数将返回void,因为累加器是原位更新,即函数的执行改变了它的内部状态以体现遍历的元素的效果。对于ToListCollector,这个函数仅仅会把当前项目添加至已经遍历过的项目的列表:
public BiConsumer<List<T>, T> accumulator() {
return (list, item) -> list.add(item);
}
3. 对结果容器应用最终转换: finisher方法在遍历完流后, finisher方法必须返回在累积过程的最后要调用的一个函数,以便将累加器对象转换为整个集合操作的最终结果。通常,就像ToListCollector的情况一样,累加器对象恰好符合预期的最终结果,因此无需进行转换。所以finisher方法只需返回identity函数:
public Function<List<T>, List<T>> finisher() {
return Function.identity();
}
四个方法中的最后一个——combiner方法会返回一个供归约操作使用的函数,它定义了对流的各个子部分进行并行处理时,各个子部分归约所得的累加器要如何合并。对于toList而言,这个方法的实现非常简单,只要把从流的第二个部分收集到的项目列表加到遍历第一部分时得到的列表后面就行了:
public BinaryOperator<List<T>> combiner() {
return (list1, list2) -> {
list1.addAll(list2);
return list1; }
}
![](https://i-blog.csdnimg.cn/blog_migrate/9451eb8cb261793603e4094bb2b248c9.png)
![](https://i-blog.csdnimg.cn/blog_migrate/fb070649f29cd1819589e9cd9a24a042.png)