Collections.frequency方法的应用
时间:2025-10-16 22:30:18 387浏览 收藏
“纵有疾风来,人生不言弃”,这句话送给正在学习文章的朋友们,也希望在阅读本文《Collections.frequency方法的应用》后,能够真的帮助到大家。我也会在后续的文章中,陆续更新文章相关的技术文章,有好的建议欢迎大家在评论留言,非常感谢!
Collections.frequency用于统计集合中指定元素的出现次数,基于equals方法比较,适用于快速、简洁地统计单个元素频次,尤其在代码可读性和维护性上优势明显。

Collections.frequency方法的核心作用,就是快速统计一个集合(Collection)中某个特定元素出现的次数。它提供了一种简洁、内置的方式来完成这项常见的计数任务,省去了我们手动遍历集合并维护计数器的麻烦。
说起Collections.frequency,它在Java标准库里算是个小巧但挺实用的工具。我记得刚开始写Java那会儿,要是想知道一个列表里某个字符串出现了多少次,最直观的做法就是写个for循环,然后里面一个if判断,匹配上了就count++。后来接触到Collections工具类,发现它竟然直接提供了frequency方法,当时就觉得“哦,原来还有这种更优雅的写法啊”。
它的用法其实很简单:
int count = Collections.frequency(collection, object);
第一个参数是你想要统计的那个Collection,比如ArrayList、LinkedList或者HashSet(尽管在HashSet里统计频率通常没啥意义,因为元素不重复,但语法上是允许的)。第二个参数就是你想要统计出现次数的那个object。
举个例子吧,假设你有一个字符串列表,想知道“apple”出现了几次:
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
public class FrequencyExample {
public static void main(String[] args) {
List<String> fruits = new ArrayList<>();
fruits.add("apple");
fruits.add("banana");
fruits.add("apple");
fruits.add("orange");
fruits.add("apple");
int appleCount = Collections.frequency(fruits, "apple");
System.out.println("Apple appears: " + appleCount + " times."); // 输出 3
}
}是不是比手写循环要清爽很多?这个方法内部其实也是遍历,但它把这些细节封装起来了,让我们的代码更专注于业务逻辑,而不是重复性的计数实现。而且,它在比较元素时用的是equals()方法,这很重要。这意味着如果你自定义了对象,并重写了equals(),它会按照你的逻辑去判断“相等”。如果没重写,那默认就是比较对象的内存地址了,这在大多数场景下可能不是你想要的。
Collections.frequency与手动循环计数:性能与场景考量
我们常常会纠结,到底是直接用Collections.frequency好,还是自己写个for循环来计数更好?这其实不是一个非黑即白的问题,它涉及到性能、代码可读性以及具体的应用场景。
从性能上看,Collections.frequency的底层实现,对于List这类有序集合,它就是简单地遍历整个集合,时间复杂度是O(n),其中n是集合的大小。这和你自己写一个for循环去遍历计数,在渐进时间复杂度上是完全一致的。所以,对于单个元素的频率统计,性能差异微乎其微,甚至可以说忽略不计。但如果你需要统计集合中 所有不同元素 的频率,那么Collections.frequency就不是最佳选择了。因为它每次调用都是完整遍历一次,如果你有m个不同的元素要统计,那总复杂度就是O(m*n),这显然效率不高。
在这种需要统计所有元素频率的场景下,一个更高效的做法是使用HashMap或者Java 8的Stream API配合Collectors.groupingBy和Collectors.counting。例如:
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.function.Function;
import java.util.stream.Collectors;
public class StreamFrequencyExample {
public static void main(String[] args) {
List<String> fruits = Arrays.asList("apple", "banana", "apple", "orange", "apple");
Map<String, Long> frequencyMap = fruits.stream()
.collect(Collectors.groupingBy(Function.identity(), Collectors.counting()));
System.out.println(frequencyMap); // {banana=1, orange=1, apple=3}
}
}这个方法的复杂度是O(n),因为它只遍历了一次集合。
所以,我的个人观点是:
- 当只需要统计集合中 一个或少数几个特定元素 的出现次数时,
Collections.frequency是首选。 它代码简洁,意图明确,可读性极佳。 - 当需要统计集合中 所有不同元素 的出现次数,或者需要更复杂的聚合操作时,考虑使用
HashMap手动构建频率映射,或者利用StreamAPI。 这样能获得更好的整体性能。 - 对于非常大的集合,如果性能是极致的瓶颈,并且你只需要判断某个元素是否存在,而不是精确计数,
HashSet的contains方法会更快(O(1)平均)。 但这和计数不是一个目的。
处理Null值和自定义对象:Collections.frequency的边界与注意事项
在使用Collections.frequency时,有几个点是需要特别留意的,尤其是涉及到null值和自定义对象时。
首先是null值。是的,Collections.frequency是可以用来统计null在集合中出现的次数的。只要你的集合允许存储null(比如ArrayList),并且你传入的object参数就是null,它就能正常工作:
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
public class NullFrequencyExample {
public static void main(String[] args) {
List<String> items = new ArrayList<>();
items.add("item1");
items.add(null);
items.add("item2");
items.add(null);
int nullCount = Collections.frequency(items, null);
System.out.println("Null appears: " + nullCount + " times."); // 输出 2
}
}这在某些数据清洗或校验的场景下,可能会派上用场,比如你想知道有多少条记录是缺失了某个字段的。
然后是自定义对象。前面提到过,Collections.frequency在比较元素时,依赖的是元素的equals()方法。如果你有一个自定义的类,比如Person:
class Person {
String name;
int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
// 省略getter/setter,为了简洁,这里不写,但实际项目中应该有
// 注意:这里没有重写equals()和hashCode()
}如果你不重写equals()和hashCode()方法,那么默认的equals()方法会比较两个对象的内存地址。这意味着,即使两个Person对象的name和age都一样,但如果它们是不同的实例,Collections.frequency也会认为它们是不同的。
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
public class CustomObjectFrequencyExample {
public static void main(String[] args) {
List<Person> people = new ArrayList<>();
people.add(new Person("Alice", 30));
people.add(new Person("Bob", 25));
people.add(new Person("Alice", 30)); // 逻辑上和第一个Alice相同,但内存地址不同
int aliceCount = Collections.frequency(people, new Person("Alice", 30));
// 这里输出 0 是大概率事件,因为 new Person("Alice", 30) 又创建了一个新的对象实例,
// 它的内存地址肯定和集合里的实例不同。
System.out.println("Alice appears (without equals/hashCode): " + aliceCount + " times.");
}
}这里输出0是大概率事件,因为new Person("Alice", 30)又创建了一个新的对象实例,它的内存地址肯定和集合里的实例不同。
为了让Collections.frequency能够正确地识别“相同”的自定义对象,你必须重写equals()和hashCode()方法。这是一个Java编程的基本约定,尤其是在将对象放入集合(特别是HashMap、HashSet)时。
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Objects; // 导入Objects工具类,方便生成equals和hashCode
class PersonWithEqualsHashCode {
String name;
int age;
public PersonWithEqualsHashCode(String name, int age) {
this.name = name;
this.age = age;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
PersonWithEqualsHashCode person = (PersonWithEqualsHashCode) o;
return age == person.age && Objects.equals(name, person.name);
}
@Override
public int hashCode() {
return Objects.hash(name, age);
}
// 省略getter/setter
}
public class CustomObjectFrequencyWithEqualsHashCodeExample {
public static void main(String[] args) {
List<PersonWithEqualsHashCode> people = new ArrayList<>();
people.add(new PersonWithEqualsHashCode("Alice", 30));
people.add(new PersonWithEqualsHashCode("Bob", 25));
people.add(new PersonWithEqualsHashCode("Alice", 30)); // 逻辑上和第一个Alice相同
int aliceCount = Collections.frequency(people, new PersonWithEqualsHashCode("Alice", 30));
System.out.println("Alice appears (with equals/hashCode): " + aliceCount + " times."); // 输出 2
}
}现在,Collections.frequency就能正确地识别出两个逻辑上相同的“Alice”了。这不仅仅是frequency方法的问题,而是所有依赖对象相等性判断的Java集合操作的共同点。
Collections.frequency在实际项目中的高级应用案例
虽然Collections.frequency看起来简单,但在某些实际场景下,它能提供非常简洁有效的解决方案。我来分享几个我个人觉得比较有意思的应用场景。
1. 快速判断元素是否出现特定次数: 有时候我们不只是想知道出现多少次,而是想知道它是否“恰好”出现了X次,或者“至少”出现了X次。比如,在一个投票系统中,你想知道某个选项是否获得了至少10票:
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
public class VoteCountExample {
public static void main(String[] args) {
List<String> votes = Arrays.asList("optionA", "optionB", "optionA", "optionC", "optionA",
"optionA", "optionB", "optionA", "optionC", "optionA",
"optionA", "optionB", "optionA"); // optionA 出现了 9 次
int requiredVotes = 10;
if (Collections.frequency(votes, "optionA") >= requiredVotes) {
System.out.println("Option A has enough votes to pass.");
} else {
System.out.println("Option A needs more votes. Currently: " + Collections.frequency(votes, "optionA"));
}
}
}这种场景下,frequency的直观性就体现出来了。
2. 集合的简单差异分析(非精确):
这可能有点“歪用”,但偶尔我会用它来做一些非常粗略的集合内容对比。比如,你有两个列表listA和listB,你想快速知道listA中某个元素X出现的次数,和listB中X出现的次数是否一致。如果数量不一致,那这两个列表在某种程度上肯定有差异。
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
public class ListDifferenceExample {
public static void main(String[] args) {
List<String> listA = Arrays.asList("apple", "banana", "apple");
List<String> listB = Arrays.asList("apple", "orange", "apple");
String target = "apple";
if (Collections.frequency(listA, target) != Collections.frequency(listB, target)) {到这里,我们也就讲完了《Collections.frequency方法的应用》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于java,php,编程的知识点!
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
164 收藏
-
341 收藏
-
125 收藏
-
427 收藏
-
152 收藏
-
129 收藏
-
334 收藏
-
431 收藏
-
294 收藏
-
292 收藏
-
183 收藏
-
288 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习