高效算法中的优先队列甄选

文章标题:

优先队列在算法问题中的巧妙应用

目录

  • 一、1046.最后的石头重量计算
  • 二、703. 数据流里的第 K 大元素
  • 三、692. 前 K 个高频词语
  • 四、295. 数据流的中位数求解

一、1046.最后的石头重量计算

题目链接:1046.最后的石头重量
题目描述:
![

](https://i-blog.csdnimg.cn/direct/16f199517b0d47498355823b0f1d8334.png)

题目解析:

  • 该问题要求我们从给定的数组中不断取出最大的两个元素,让它们相减后将差值重新放入数组,重复此过程直到数组为空或只剩一个元素。

解题思路:

  • 直接对数组排序会频繁操作,效率不高。此时可使用优先队列(大根堆),能快速取出最大元素,降低操作开销。

解题代码:

//时间复杂度 O(n)
//空间复杂度 O(n)
class Solution {
    public int lastStoneWeight(int[] stones) {
        //创建大根堆
        PriorityQueue<Integer> queue = new PriorityQueue<>(
            (a,b) -> b - a
        );
        //将数组元素加入堆中
        for(int i = 0; i < stones.length; i++)
            queue.offer(stones[i]);
        //循环处理,直到堆中只剩一个或没有元素
        while(!queue.isEmpty() && queue.size() != 1) {
            int y = queue.poll();
            int x = queue.poll();
            queue.offer(y-x); 
        }
        //返回结果,如果堆为空返回0,否则返回剩余元素
        return queue.isEmpty() ? 0 : queue.poll();
    }
}

二、703. 数据流里的第 K 大元素

题目链接:703. 数据流中的第 K 大元素

题目描述:
![

](https://i-blog.csdnimg.cn/direct/7b82ee541d5a4890b4ebe3b6e3473fea.png)

题目解析:

  • 给定一个数组和整数K,需通过类的add方法,每次添加元素后返回当前数组中第K大的元素。

解题思路:

  • 利用大小为K的小根堆,堆中始终保存当前数组中第K大到最大的元素,堆顶即为第K大元素。

解题代码:

//时间复杂度 O(nLogK)
//空间复杂度 O(K)
class KthLargest {
    PriorityQueue<Integer> heap;
    int kValue;

    public KthLargest(int k, int[] nums) {
        heap = new PriorityQueue<Integer>();
        kValue = k;
        for(int i = 0; i < nums.length; i++) {
            heap.offer(nums[i]);
            if(heap.size() > kValue) {
                heap.poll();
            }
        }

    }

    public int add(int val) {
        heap.offer(val);
        if(heap.size() > kValue) {
            heap.poll();
        }
        return heap.peek();
    }
}

三、692. 前 K 个高频词语

题目链接:692. 前 K 个⾼频单词

题目描述:
![

](https://i-blog.csdnimg.cn/direct/9867706b13004912a11605b9a5555c3b.png)

题目解析:

  • 给定字符串数组words,需找出出现频率最高的前K个字符串,频率相同时按字典序从小到大排列。

解题思路:

  • 先用哈希表统计字符串出现次数,再用大小为K的堆,结合自定义比较器处理频率和字典序,最后逆序结果得到正确顺序。

解题代码:

//时间复杂度:O(NLogK)
//空间复杂度:O(N)
class Solution {
    public List<String> topKFrequent(String[] words, int k) {
        Map<String, Integer> countMap = new HashMap<>();

        PriorityQueue<Pair<String,Integer>> heap = new PriorityQueue<>(
            (a,b) -> {
            //频率相同时按字典序降序,保证弹出时为升序
            if(a.getValue().equals(b.getValue())) {
                return b.getKey().compareTo(a.getKey());
            }
            //频率不同时按升序
            return a.getValue() - b.getValue();
        });
        //统计单词出现次数
        for( String s : words) {
            countMap.put(s, countMap.getOrDefault(s ,0) + 1);
        }
        //将元素放入堆,保持堆大小不超k
        for(Map.Entry<String, Integer> entry : countMap.entrySet()) {
            heap.offer(new Pair<>(entry.getKey(),entry.getValue()));
            if(heap.size() > k) {
                heap.poll();
            }
        }
        //收集结果并逆序
        List<String> result = new ArrayList<String>();
        while(!heap.isEmpty()) {
            result.add(heap.poll().getKey());
        }
        Collections.reverse(result);
        return result;
    } 
}

四、295. 数据流的中位数求解

题目链接:295. 数据流的中位数

题目描述:

  • 需实现一个类,能初始化、添加元素,并在添加后快速获取数据流的中位数。

题目解析:

  • 通过维护大根堆和小根堆,根据元素个数调整堆的分布,从而快速得到中位数。

解题思路:

  • 维护大根堆(存前半部分元素)和小根堆(存后半部分元素),根据插入元素的大小和堆的情况调整元素分布,保证元素个数为偶数时两堆平衡,奇数时一堆多一个元素。

解题代码:

//时间复杂度:O(LogN)
//空间复杂度:O(N)
class MedianFinder {
    //记录元素个数
    int elementCount = 0;
    //大根堆存储前半部分元素
    PriorityQueue<Integer> maxHeap;
    //小根堆存储后半部分元素
    PriorityQueue<Integer> minHeap;


    public MedianFinder() {
        maxHeap = new PriorityQueue<>((a,b) ->{
            return b - a;
        });
        minHeap = new PriorityQueue<>();
    }

    public void addNum(int num) {
        elementCount++;
        if(elementCount == 1 ) {
            maxHeap.offer(num);
            return;
        }
        //元素个数为偶数且当前数大于大根堆堆顶
        if(elementCount % 2 == 0 && maxHeap.peek() <= num) {
            if(maxHeap.size() < minHeap.size()) {
                if(!minHeap.isEmpty() && minHeap.peek() >= num) {
                    maxHeap.offer(num);
                }else {
                    int temp = minHeap.poll();
                    maxHeap.offer(temp);
                    minHeap.offer(num);
                }
            }else {
                minHeap.offer(num);
            }
            return;
        } 
         //元素个数为偶数且当前数小于大根堆堆顶
        if(elementCount % 2 == 0 && maxHeap.peek() > num) {
            if(maxHeap.size() < minHeap.size()) {
                maxHeap.offer(num);
            }else {
                int temp = maxHeap.poll();
                maxHeap.offer(num);
                minHeap.offer(temp);

            }
            return;
        } 
         //元素个数为奇数且当前数小于大根堆堆顶
         if(elementCount % 2 != 0 && maxHeap.peek() >= num) {
            maxHeap.offer(num);
         } else {
            minHeap.offer(num);
         }
    }

    public double findMedian() {
        if(elementCount % 2 == 0) {
                return (double)((maxHeap.peek() + minHeap.peek())/ 2.0);
        }
        if(maxHeap.size() > minHeap.size()) {
             return maxHeap.peek();
        } else {
            return minHeap.peek();
        }

    }
}

文章整理自互联网,只做测试使用。发布者:Lomu,转转请注明出处:https://www.it1024doc.com/12847.html

(0)
LomuLomu
上一篇 2025 年 7 月 9 日
下一篇 2025 年 7 月 10 日

相关推荐

  • 永久IDEA激活码下载及IDEA破解经验

    重要提示:以下教程中的 IntelliJ IDEA 破解补丁、激活码均源自网络公开渠道,仅供个人学习研究,禁止商业用途。如涉侵权,请联系删除;条件允许请支持正版! IntelliJ IDEA 是 JetBrains 打造的跨平台 IDE,Windows、macOS、Linux 通吃。下文手把手教你用破解补丁一键永久激活,解锁全部高级特性。 无论你装的是哪个版…

    IDEA破解教程 2025 年 12 月 3 日
    11300
  • 促销系统:促销活动、优惠券、优惠规则概念模型设计

    大家好,我是汤师爷~ 概念模型设计是促销系统开发的关键环节,我们需要基于之前的功能分析,将复杂的促销业务拆解成清晰的领域概念,这些概念之间的关系界定和边界划分,将直接决定系统的可维护性和扩展性。 促销系统核心概念模型 通过对促销业务的分析,我们可以抽象出促销系统的关键概念模型。 1、促销活动模型 促销活动模型对活动的各个要素和规则进行抽象,包含活动名称、描述…

    2025 年 1 月 12 日
    51100
  • 全网资源同步2025最新版webstorm激活码,权威破解教程

    申明:本教程 WebStorm破解补丁、激活码均收集于网络,请勿商用,仅供个人学习使用,如有侵权,请联系作者删除。若条件允许,希望大家购买正版 ! 废话不多说,先上 WebStorm 2025.2.1 版本破解成功的截图,如下图,可以看到已经成功破解到 2099 年辣,舒服的很! 接下来就给大家通过图文的方式分享一下如何破解最新的WebStorm。 如果觉得…

    2025 年 12 月 30 日
    17800
  • 2025年最新IDEA激活码永久破解教程 | IDEA注册码及破解方法详解

    JetBrains全家桶通用破解指南(支持IDEA/PyCharm/DataGrip等) 先给大家展示最新IDEA版本成功破解的截图,可以看到有效期已延长至2099年! 下面将详细介绍如何实现IDEA的长期激活,本方法同样适用于旧版本,无论您使用什么操作系统或版本都能适用。 获取IDEA安装包 若已安装可跳过此步骤 访问JetBrains官网:https:/…

    IDEA破解教程 2025 年 7 月 19 日
    20300
  • 多端适配webstorm激活码获取渠道,最全webstorm破解教程同步

    申明:本教程 WebStorm破解补丁、激活码均收集于网络,请勿商用,仅供个人学习使用,如有侵权,请联系作者删除。若条件允许,希望大家购买正版 ! 废话不多说,先上 WebStorm 2025.2.1 版本破解成功的截图,如下图,可以看到已经成功破解到 2099 年辣,舒服的很! 接下来就给大家通过图文的方式分享一下如何破解最新的WebStorm。 如果觉得…

    2025 年 12 月 28 日
    8300

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

联系我们

400-800-8888

在线咨询: QQ交谈

邮件:admin@example.com

工作时间:周一至周五,9:30-18:30,节假日休息

关注微信