Collections.frequency用于统计集合中某元素出现次数,其依赖equals方法判断相等,处理自定义对象需重写equals和hashCode,性能为O(n),适用于小到中型集合;对于大数据量或频繁查询场景,使用HashMap或Stream API构建频率映射更高效,可将查询时间降为O(1)。

Collections.frequency
Collection
Java.util.Collections
frequency(Collection<?> c, Object o)
c
o.equals(element)
element
o
这个方法设计得非常直观。比如你有一个字符串列表,想知道“Apple”出现了几次,直接调用
Collections.frequency(myList, "Apple")
值得注意的是,它依赖于对象的
equals()
equals()
frequency
null
null
null
null
立即学习“Java免费学习笔记(深入)”;
从性能角度看,
Collections.frequency
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");
fruits.add(null);
fruits.add("Banana");
fruits.add(null);
// 统计 "Apple" 出现的次数
int appleCount = Collections.frequency(fruits, "Apple");
System.out.println("Apple 出现的次数: " + appleCount); // 输出 3
// 统计 "Banana" 出现的次数
int bananaCount = Collections.frequency(fruits, "Banana");
System.out.println("Banana 出现的次数: " + bananaCount); // 输出 2
// 统计 "Grape" 出现的次数 (不存在)
int grapeCount = Collections.frequency(fruits, "Grape");
System.out.println("Grape 出现的次数: " + grapeCount); // 输出 0
// 统计 null 出现的次数
int nullCount = Collections.frequency(fruits, null);
System.out.println("null 出现的次数: " + nullCount); // 输出 2
// 尝试统计一个不存在的自定义对象
class MyObject {
String name;
MyObject(String name) { this.name = name; }
// 没有重写 equals() 和 hashCode()
}
List<MyObject> myObjects = new ArrayList<>();
myObjects.add(new MyObject("A"));
myObjects.add(new MyObject("B"));
myObjects.add(new MyObject("A"));
int myObjectACount = Collections.frequency(myObjects, new MyObject("A"));
System.out.println("MyObject(\"A\") 出现的次数 (未重写 equals): " + myObjectACount); // 输出 0,因为是不同对象引用
}
}Collections.frequency
在我看来,
Collections.frequency
for
手动遍历计数通常是这样的:
public static int countManually(List<String> list, String target) {
int count = 0;
for (String item : list) {
if (target == null ? item == null : target.equals(item)) {
count++;
}
}
return count;
}而
Collections.frequency
至于性能,坦白说,对于大多数日常应用场景,两者的差异微乎其微,几乎可以忽略不计。它们都是 O(n) 的时间复杂度,这意味着它们的执行时间会随着集合大小的线性增长。
Collections.frequency
我个人更倾向于使用
Collections.frequency
Collections.frequency
Collections.frequency
处理
null
Collections.frequency
null
null
null
List<String> items = new ArrayList<>();
items.add("A");
items.add(null);
items.add("B");
items.add(null);
int nullCount = Collections.frequency(items, null); // nullCount 为 2
System.out.println("null 元素的数量: " + nullCount);但是,当涉及到自定义对象时,事情就有点儿意思了,这里面的“坑”主要集中在
equals()
Collections.frequency
o.equals(element)
如果你的自定义类没有重写
equals()
Object
equals()
obj1 == obj2
Collections.frequency
class Product {
String name;
double price;
public Product(String name, double price) {
this.name = name;
this.price = price;
}
// 没有重写 equals()
}
List<Product> products = new ArrayList<>();
products.add(new Product("Laptop", 1200.0));
products.add(new Product("Mouse", 25.0));
products.add(new Product("Laptop", 1200.0)); // 另一个实例,但属性相同
// 尝试查找一个属性相同的 Product 对象
int laptopCount = Collections.frequency(products, new Product("Laptop", 1200.0));
System.out.println("Laptop 出现的次数 (未重写 equals): " + laptopCount); // 结果通常是 0要让
Collections.frequency
equals()
equals()
hashCode()
frequency
hashCode()
HashMap
HashSet
class ProductWithEquals {
String name;
double price;
public ProductWithEquals(String name, double price) {
this.name = name;
this.price = price;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
ProductWithEquals that = (ProductWithEquals) o;
return Double.compare(that.price, price) == 0 &&
name.equals(that.name);
}
@Override
public int hashCode() {
return Objects.hash(name, price);
}
}
List<ProductWithEquals> productsWithEquals = new ArrayList<>();
productsWithEquals.add(new ProductWithEquals("Laptop", 1200.0));
productsWithEquals.add(new ProductWithEquals("Mouse", 25.0));
productsWithEquals.add(new ProductWithEquals("Laptop", 1200.0)); // 另一个实例,但属性相同
int laptopCountCorrect = Collections.frequency(productsWithEquals, new ProductWithEquals("Laptop", 1200.0));
System.out.println("Laptop 出现的次数 (重写 equals): " + laptopCountCorrect); // 结果现在是 2所以,在使用
Collections.frequency
equals()
hashCode()
Collections.frequency
Collections.frequency
我个人觉得,最常用的高效计数方法是使用
Map
Map
Map
import java.util.HashMap;
import java.util.Map;
import java.util.List;
import java.util.Arrays;
import java.util.function.Function;
import java.util.stream.Collectors;
// ... (假设有前面的 List<String> fruits 集合)
public class EfficientCountingExample {
public static void main(String[] args) {
List<String> fruits = Arrays.asList("Apple", "Banana", "Apple", "Orange", "Apple", null, "Banana", null);
// 方法一:使用 HashMap 手动计数
Map<String, Integer> frequencyMap = new HashMap<>();
for (String fruit : fruits) {
frequencyMap.put(fruit, frequencyMap.getOrDefault(fruit, 0) + 1);
}
System.out.println("HashMap 手动计数结果: " + frequencyMap);
// 之后查询 "Apple" 的次数就是 O(1)
System.out.println("Apple 出现次数 (Map查询): " + frequencyMap.get("Apple"));
// 方法二:使用 Java 8 Stream API
// 注意:Stream API 默认不会将 null 作为 key,或者需要特殊处理
Map<String, Long> streamFrequencyMap = fruits.stream()
.filter(f -> f != null) // 过滤掉 null,或者根据需求处理
.collect(Collectors.groupingBy(Function.identity(), Collectors.counting()));
System.out.println("Stream API 计数结果 (不含null): " + streamFrequencyMap);
System.out.println("Apple 出现次数 (Stream Map查询): " + streamFrequencyMap.get("Apple"));
// 如果要包含 null,可以这样处理:
Map<String, Long> streamFrequencyMapWithNull = fruits.stream()
.collect(Collectors.groupingBy(f -> f == null ? "NULL_KEY" : f, Collectors.counting()));
System.out.println("Stream API 计数结果 (含null,用特殊键表示): " + streamFrequencyMapWithNull);
System.out.println("null 出现次数 (Stream Map查询): " + streamFrequencyMapWithNull.get("NULL_KEY"));
}
}HashMap 手动计数:
Map
Collections.frequency
Map
Java 8 Stream API:
null
filter
null
null
选择建议:
Collections.frequency
HashMap
HashMap
在我看来,掌握这几种计数方法,能让你在不同的场景下灵活选择最合适的工具,写出更高效、更健壮的Java代码。很多时候,选择不是非黑即白,而是根据实际需求和上下文来做权衡。
以上就是Java中Collections.frequency方法解析的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号