0

0

Java Stream 高效分组计数并获取Top N元素

碧海醫心

碧海醫心

发布时间:2025-10-23 11:18:01

|

402人浏览过

|

来源于php中文网

原创

Java Stream 高效分组计数并获取Top N元素

本文深入探讨了如何利用java stream api对数据进行高效的分组计数,并从中提取出现频率最高的top n元素。文章首先介绍了一种简洁的基于全排序的实现方式,该方法适用于数据集较小或top n值接近总数的情况。随后,针对大数据量和小型top n场景下的性能瓶颈,文章详细阐述了如何通过自定义`collector`结合`priorityqueue`(最小堆)进行部分排序,以显著提升处理效率,并提供了完整的代码示例及详细的实现原理。

Java Stream API为集合数据的处理提供了强大而富有表达力的工具。在实际开发中,我们经常会遇到需要对数据进行分组统计,并找出其中频率最高(或最低)的Top N元素的需求。例如,统计全球城市数据中,拥有城市数量最多的Top 3国家。本文将介绍两种实现这一目标的Stream方法,并分析它们的适用场景和性能特点。

方法一:基于全排序的简洁实现

这种方法的核心思想是:首先利用Collectors.groupingBy对数据进行分组,并通过Collectors.counting()计算每个分组的元素数量。这将生成一个Map,其中键K是分组字段,值Long是对应的计数。然后,将这个Map的entrySet转换为Stream,并根据值(计数)进行降序排序,最后通过limit()操作获取前N个元素,并提取其键。

示例代码

假设我们有一个City实体类,包含countryCode字段,我们希望找出拥有城市数量最多的Top N个国家代码:

import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;

// 假设 City 类定义如下
class City {
    private int id;
    private String name;
    private String countryCode;

    public City(int id, String name, String countryCode) {
        this.id = id;
        this.name = name;
        this.countryCode = countryCode;
    }

    public String getCountryCode() {
        return countryCode;
    }

    @Override
    public String toString() {
        return "City{" + "id=" + id + ", name='" + name + '\'' + ", countryCode='" + countryCode + '\'' + '}';
    }
}

public class StreamTopNMethods {

    /**
     * 使用全排序获取 Top N 国家代码
     * @param cities 城市列表
     * @param limit 需要获取的 Top N 数量
     * @return Top N 国家代码列表
     */
    public static List getTopNCodesByFullSort(List cities, int limit) {
        return cities.stream()
            .collect(Collectors.groupingBy( // 1. 按 countryCode 分组并计数
                City::getCountryCode,
                Collectors.counting()
            )) // 结果为 Map,例如 {DE=4, FR=2, DK=1, NO=1}
            .entrySet().stream() // 2. 获取 Map 的 entrySet 并转换为 Stream
            .sorted(Map.Entry.comparingByValue().reversed()) // 3. 按计数降序排序
            .limit(limit) // 4. 取前 limit 个元素
            .map(Map.Entry::getKey) // 5. 提取国家代码 (键)
            .toList(); // 6. 收集结果到 List
    }

    public static void main(String[] args) {
        List cityList = List.of(
            new City(1, "Berlin", "DE"),
            new City(2, "Munich", "DE"),
            new City(3, "Köln", "DE"),
            new City(4, "Paris", "FR"),
            new City(5, "Kopenhag", "DK"),
            new City(6, "Hamburg", "DE"),
            new City(7, "Lyon", "FR"),
            new City(8, "Oslo", "NO"),
            new City(9, "Frankfurt", "DE")
        );

        List top3Countries = getTopNCodesByFullSort(cityList, 3);
        System.out.println("Top 3 Countries by city count (Full Sort): " + top3Countries);
        // 示例输出: Top 3 Countries by city count (Full Sort): [DE, FR, DK] (或 [DE, FR, NO],取决于DK和NO在Map中的相对顺序,因为它们的计数相同)
    }
}

性能分析与注意事项

这种方法的优点是代码简洁、易于理解。然而,它的时间复杂度为 O(M log M),其中 M 是分组后得到的唯一键的数量。这是因为 sorted() 操作会对整个 entrySet 进行排序。当原始数据集非常庞大,导致 M 很大,而我们只需要获取少数几个Top N元素时(即 limit 远小于 M),这种全局排序的开销会比较大,效率较低。

立即学习Java免费学习笔记(深入)”;

一键职达
一键职达

AI全自动批量代投简历软件,自动浏览招聘网站从海量职位中用AI匹配职位并完成投递的全自动操作,真正实现'一键职达'的便捷体验。

下载

方法二:利用自定义 Collector 和 PriorityQueue 进行部分排序

为了优化上述方法的性能瓶颈,我们可以采用一种更高效的策略:利用最小堆(PriorityQueue)实现部分排序。这种方法避免了对所有分组结果进行完全排序,而是在遍历过程中动态维护一个包含Top N元素的堆。当数据集庞大且 N 值相对较小时,这种方法能显著提升性能。

核心思想

首先,与方法一相同,我们通过Collectors.groupingBy得到 Map。然后,我们创建一个自定义的Collector,其内部使用一个大小为 N 的 PriorityQueue。这个PriorityQueue被配置为一个最小堆,用于存储当前遍历到的Top N元素。每当处理一个新的Map entry时,我们将其值(计数)与堆中最小元素的值进行比较:如果新元素的值更大,则移除堆中最小元素,并添加新元素。

泛化方法与自定义 Collector 实现

为了代码的复用性,我们可以将获取Top N的逻辑泛化。

import java.util.Comparator;
import java.util.List;
import java.util.Map;
import java.util.PriorityQueue;
import java.util.Queue;
import java.util.function.Function;
import java.util.stream.Collector;
import java.util.stream.Collectors;

// ... City 类定义同上

public class StreamTopNMethodsAdvanced {

    /**
     * 泛化方法:使用自定义 Collector 和 PriorityQueue 获取 Top N 元素
     * @param list 原始数据列表
     * @param keyExtractor 用于从原始数据中提取分组键的函数
     * @param limit 需要获取的 Top N 数量
     * @param  原始数据类型
     * @param  分组键类型
     * @return Top N 键的列表
     */
    public static  List getTopNByPriorityQueue(List list,
                                                         Function keyExtractor,
                                                         int limit) {
        return list.stream()
            .collect(Collectors.groupingBy( // 1. 按 keyExtractor 提取的键分组并计数
                keyExtractor,
                Collectors.counting()
            )) // 结果为 Map
            .entrySet().stream() // 2. 获取 Map 的 entrySet 并转换为 Stream
            .collect(getMaxNCollector( // 3. 使用自定义 Collector 获取 Top N
                limit,
                Map.Entry.comparingByValue().reversed(), // 比较器,用于 PriorityQueue 维护最小堆
                Map.Entry::getKey // 提取最终结果的函数
            ));
    }

    /**
     * 辅助方法:向 PriorityQueue 中尝试添加元素
     * 确保队列始终只包含 'size' 个元素,且这些元素是当前遍历过所有元素中“最大”的 'size' 个。
     * @param queue PriorityQueue 实例
     * @param next 待添加的下一个元素
     * @param comparator 用于比较元素的比较器
     * @param size 队列的最大容量 (即 N 值)
     * @param  队列中元素的类型
     */
    public static  void tryAdd(Queue queue, T next, Comparator comparator, int size) {
        // 如果队列已满,且新元素比队列中“最小”的元素(即堆顶元素)“更大”(根据比较器)
        // 注意:这里的 comparator 是 reversed 的,所以 compare(queue.element(), next) < 0 意味着 next 更大
        if (queue.size() == size && comparator.compare(queue.element(), next) < 0) {
            queue.remove(); // 移除堆顶元素(当前 Top N 中最小的那个)
        }
        // 如果队列未满,或者新元素被认为“更大”并已移除旧元素,则添加新元素
        if (queue.size() < size) {
            queue.add(next);
        }
    }

    /**
     * 自定义 Collector,用于从 Stream 中获取 Top N 元素。
     * @param size 需要获取的 Top N 数量
     * @param comparator 用于在 PriorityQueue 中排序元素的比较器。
     *                   如果需要 Top N 最大值,此比较器应使得“较小”的元素排在前面(即为最小堆)。
     *                   对于 Map.Entry 场景,若要获取计数最大的 N 个,则比较器应为 Map.Entry.comparingByValue()。
     *                   但由于 PriorityQueue 默认是最小堆,为了让“最大”的元素留在堆中,我们传入一个“反向”的比较器。
     *                   例如,`Map.Entry.comparingByValue().reversed()`,这样计数小的元素会在堆顶。
     * @param keyExtractor 用于从堆中的元素(Map.Entry)提取最终结果(键)的函数
     * @param  Stream 中元素的类型 (这里是 Map.Entry					

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

844

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

743

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

740

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

400

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

447

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

431

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

c++空格相关教程合集
c++空格相关教程合集

本专题整合了c++空格相关教程,阅读专题下面的文章了解更多详细内容。

0

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.8万人学习

C# 教程
C# 教程

共94课时 | 7.4万人学习

Java 教程
Java 教程

共578课时 | 50.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号