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<String> getTopNCodesByFullSort(List<City> cities, int limit) {
        return cities.stream()
            .collect(Collectors.groupingBy( // 1. 按 countryCode 分组并计数
                City::getCountryCode,
                Collectors.counting()
            )) // 结果为 Map<String, Long>,例如 {DE=4, FR=2, DK=1, NO=1}
            .entrySet().stream() // 2. 获取 Map 的 entrySet 并转换为 Stream
            .sorted(Map.Entry.<String, Long>comparingByValue().reversed()) // 3. 按计数降序排序
            .limit(limit) // 4. 取前 limit 个元素
            .map(Map.Entry::getKey) // 5. 提取国家代码 (键)
            .toList(); // 6. 收集结果到 List
    }

    public static void main(String[] args) {
        List<City> 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<String> 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),这种全局排序的开销会比较大,效率较低。

人民网AIGC-X
人民网AIGC-X

国内科研机构联合推出的AI生成内容检测工具

下载

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

方法二:利用自定义 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 <T> 原始数据类型
     * @param <K> 分组键类型
     * @return Top N 键的列表
     */
    public static <T, K> List<K> getTopNByPriorityQueue(List<T> list,
                                                         Function<T, K> keyExtractor,
                                                         int limit) {
        return list.stream()
            .collect(Collectors.groupingBy( // 1. 按 keyExtractor 提取的键分组并计数
                keyExtractor,
                Collectors.counting()
            )) // 结果为 Map<K, Long>
            .entrySet().stream() // 2. 获取 Map 的 entrySet 并转换为 Stream
            .collect(getMaxNCollector( // 3. 使用自定义 Collector 获取 Top N
                limit,
                Map.Entry.<K, Long>comparingByValue().reversed(), // 比较器,用于 PriorityQueue 维护最小堆
                Map.Entry::getKey // 提取最终结果的函数
            ));
    }

    /**
     * 辅助方法:向 PriorityQueue 中尝试添加元素
     * 确保队列始终只包含 'size' 个元素,且这些元素是当前遍历过所有元素中“最大”的 'size' 个。
     * @param queue PriorityQueue 实例
     * @param next 待添加的下一个元素
     * @param comparator 用于比较元素的比较器
     * @param size 队列的最大容量 (即 N 值)
     * @param <T> 队列中元素的类型
     */
    public static <T> void tryAdd(Queue<T> queue, T next, Comparator<T> 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<K, Long> 场景,若要获取计数最大的 N 个,则比较器应为 Map.Entry.comparingByValue()。
     *                   但由于 PriorityQueue 默认是最小堆,为了让“最大”的元素留在堆中,我们传入一个“反向”的比较器。
     *                   例如,`Map.Entry.<K, Long>comparingByValue().reversed()`,这样计数小的元素会在堆顶。
     * @param keyExtractor 用于从堆中的元素(Map.Entry)提取最终结果(键)的函数
     * @param <T> Stream 中元素的类型 (这里是 Map.Entry<K, Long

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

443

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

605

2023.08.10

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

77

2025.09.05

golang map相关教程
golang map相关教程

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

40

2025.11.16

golang map原理
golang map原理

本专题整合了golang map相关内容,阅读专题下面的文章了解更多详细内容。

67

2025.11.17

java判断map相关教程
java判断map相关教程

本专题整合了java判断map相关教程,阅读专题下面的文章了解更多详细内容。

47

2025.11.27

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

76

2026.03.11

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

38

2026.03.10

Kotlin Android模块化架构与组件化开发实践
Kotlin Android模块化架构与组件化开发实践

本专题围绕 Kotlin 在 Android 应用开发中的架构实践展开,重点讲解模块化设计与组件化开发的实现思路。内容包括项目模块拆分策略、公共组件封装、依赖管理优化、路由通信机制以及大型项目的工程化管理方法。通过真实项目案例分析,帮助开发者构建结构清晰、易扩展且维护成本低的 Android 应用架构体系,提升团队协作效率与项目迭代速度。

83

2026.03.09

热门下载

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

精品课程

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

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.2万人学习

Java 教程
Java 教程

共578课时 | 81万人学习

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

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