Collectors.counting()用于统计流中元素数量,常与groupingBy结合实现分类计数,如按首字母分组统计名字个数;也可配合filter或partitioningBy进行条件计数,适用于复杂数据聚合场景。

在Java 8引入的Stream API中,Collectors.counting 是一个非常实用的收集器,用于统计流中元素的数量。它常与 groupingBy、filtering 等操作结合使用,实现灵活的数据聚合。本文将通过实际场景讲解如何正确高效地使用 Collectors.counting() 进行计数操作。
基础用法:统计集合总数量
最简单的使用方式是配合 stream().collect() 统计整个集合的元素个数。
例如,统计一个字符串列表的长度:
List
long count = names.stream().collect(Collectors.counting());
System.out.println(count); // 输出:4
注意:虽然此处可用 names.size() 更高效,但此方式展示了 Collectors.counting() 的基本语法结构,适用于更复杂的流处理链。
立即学习“Java免费学习笔记(深入)”;
分组统计:按条件分类计数
更常见的场景是结合 Collectors.groupingBy 实现分类计数,比如统计每个首字母出现的次数。
Map
.collect(Collectors.groupingBy(s -> s.charAt(0), Collectors.counting()));
System.out.println(countByFirstLetter); // 输出如:{A=1, B=1, C=1, D=1}
这里 groupingBy 按首字母分组,每个组内使用 counting() 统计元素数量,结果为 Map。
过滤后计数:条件性统计
有时需要先筛选再计数。可以使用 filter 配合 counting(),或嵌套收集器实现更精细控制。
例如,统计名字长度大于5的个数:
long longNameCount = names.stream()
.filter(name -> name.length() > 5)
.collect(Collectors.counting());
也可以在分组时加入过滤逻辑(Java 9+):
Map
.collect(Collectors.partitioningBy(s -> s.length() > 5, Collectors.counting()));
这会返回一个布尔键的Map,分别表示“是否长度大于5”的计数。
性能建议与注意事项
尽管 Collectors.counting() 功能强大,但在简单场景下需注意性能和可读性:
- 如果只是获取集合大小,直接调用
size()方法比流更高效。 - 在并行流中使用
counting()是安全的,内部基于reduce实现,支持并发合并。 - 返回类型是
Long,即使数据量小也需注意类型转换问题。 - 避免在循环中频繁创建流进行计数,应尽量合并操作或缓存结果。
基本上就这些。掌握 Collectors.counting() 的核心在于理解其作为终端收集器的角色,尤其在分组和条件聚合中的价值。合理使用能让代码更简洁且表达力更强。










