Collectors.reducing提供三种重载形式:无初始值归约、带初始值归约、映射后归约,适用于求和、最值及自定义累积,提升Stream操作灵活性与可读性。

在Java 8引入的Stream API中,Collectors.reducing 是一个强大的归约工具,用于对流中的元素进行累积操作。它适用于求和、求最大值、最小值,或自定义的合并逻辑。掌握它的用法,可以简化集合处理代码,提升可读性和功能性。
理解 Collectors.reducing 的三种重载形式
Collectors.reducing 提供了三个重载方法,适应不同场景:
-
reducing(BinaryOperator
) :最简形式,对流中所有元素执行二元操作。若流为空,返回 Optional.empty()。 -
reducing(T identity, BinaryOperator
) :指定初始值(identity),即使流为空也返回该值。 -
reducing(U identity, Function
mapper, BinaryOperator) :支持类型转换,先映射再归约,最灵活。
这些形式覆盖了大多数归约需求,从简单数值累加到复杂对象属性提取。
常见应用场景与代码示例
以下是一些典型使用方式,帮助理解其实际价值。
立即学习“Java免费学习笔记(深入)”;
1. 数值求和
Listnumbers = Arrays.asList(1, 2, 3, 4, 5); int sum = numbers.stream() .collect(Collectors.reducing(0, Integer::sum)); // 结果:15
这里使用 identity 为 0,确保空集合也能安全返回 0。
2. 获取最大值或最小值
Listwords = Arrays.asList("apple", "banana", "cherry"); Optional longest = words.stream() .collect(Collectors.reducing((a, b) -> a.length() > b.length() ? a : b)); // 返回最长字符串的 Optional
注意返回类型是 Optional,需判断是否存在结果。
3. 对象属性归约
class Product {
String name;
double price;
// 构造函数和 getter 省略
}
List products = ...;
double totalPrice = products.stream()
.collect(Collectors.reducing(0.0, Product::getPrice, Double::sum));
这是第三种形式的经典应用:将 Product 映射为价格,再进行累加。
注意事项与最佳实践
使用 reducing 时,有几个关键点需要注意:
- 归约操作应满足结合律,保证并行流下的正确性。
- 避免在归约函数中产生副作用,如修改外部变量。
- 当目标是求和、计数等常见操作时,优先使用专门收集器如 Collectors.summingDouble,更直观。
- 对于可能为空的流,考虑是否需要默认值,选择合适的重载版本。
基本上就这些。Collectors.reducing 虽然不如 groupingBy 那样常用,但在需要自定义累积逻辑时非常实用。掌握它,能让你的Stream操作更加灵活高效。










