使用PriorityQueue创建大小堆,解决TOPK问题

集合框架中的PriorityQueue底层使用堆结构,因此其内部的元素必须要能够比大小,PriorityQueue采用了: Comparble和Comparator两种方式。

  1. Comparble是默认的内部比较方式,如果用户插入自定义类型对象时,该类对象必须要实现Comparble接 口,并覆写compareTo方法

  2. 用户也可以选择使用比较器对象,如果用户插入自定义类型对象时,必须要提供一个比较器类,让该类实现 Comparator接口并覆写compare方法。

创建大小堆,先实现比较器

java 复制代码
//小根堆
class LessComp implements Comparator<Integer> {

    @Override
    public int compare(Integer o1, Integer o2) {
        return o1 - o2;
    }
}
//大根堆
class GreaterComp implements Comparator<Integer> {

    @Override
    public int compare(Integer o1, Integer o2) {
        return o2 - o1;
    }
}

例如找到最大的k个数

java 复制代码
    public int[] maxKey(int[] array, int k) {
        if(k < 0){
            return null;
        }
        LessComp gc = new LessComp();
        PriorityQueue<Integer> pq = new PriorityQueue<>(gc);
        for(int i = 0; i < k; i++){
            pq.offer(array[i]);
        }

        for(int i = k; i < array.length; i++){
            int top = pq.peek();
            if(top < array[i]){
                pq.poll();
                pq.offer(array[i]);
            }
        }
        int[] result = new int[k];
        for(int i = 0; i < k; i++){
            result[i] = pq.poll();
        }
        return result;
    }

要维护一个长度为k的小根堆,堆顶是堆里最小的数,然后开始比较,将大于堆顶的数入队,内部会进行向上调整,依次将剩下的数据处理。本例是找最大的k个数,要维护一个小根堆,反之,找最小的k个数,就维护一个大根堆。但要注意入堆的条件。

当维护小根堆的时候,只有当新数据比堆顶大时,才入堆。(最大的前k个数)

当维护大根堆的时候,只有当新数据比堆顶小时,才入堆。(最小的前k个数)

相关推荐
一嘴一个橘子10 分钟前
MP 自定义业务方法 (三)
java
一叶飘零_sweeeet11 分钟前
AI Agent 深潜:六大核心模块的设计本质与 Java 实现
java·人工智能·agent
向往着的青绿色19 分钟前
Java反序列化漏洞(持续更新中)
java·开发语言·计算机网络·安全·web安全·网络安全·网络攻击模型
小短腿的代码世界41 分钟前
Qt跨进程通信在交易系统中的应用:让策略引擎与风控模块在毫秒级握手
开发语言·qt
Carsene41 分钟前
第一章:为什么我们需要“类型安全”的 SQL DSL 框架?
java·sql
wyu7296144 分钟前
Spring MVC 学习笔记:配置、注解、RESTful、JSON、拦截器、SSM整合、文件上传下载
java
zhangrelay1 小时前
三分钟云课实践速通--大学物理--python 版
linux·开发语言·python·学习·ubuntu·lubuntu
Mr_pyx1 小时前
Java 注解(Annotation)详解:从基础到 APT 实战
java·数据库·sqlserver
MegaDataFlowers1 小时前
调用Service层操作数据
java·开发语言
asdzx672 小时前
使用 Python 读取 PDF: 提取文本和图片
开发语言·python·pdf