【优选算法】优先级队列:最后一块石头的重量,数据流中的第K大元素,前K个高频单词,数据流中的中位数

文章目录

TOP - K问题主要有两种比较优秀的解法:

  • 时间复杂度:O(n * log2n)
  • 快速选择算法 时间复杂度O(n) ,空间开销比较大

1. 最后一块石头的重量(LC1046)

最后一块石头的重量

题目描述

解题思路

利用大根堆模拟这个过程,每次取出前两个大的元素碰撞,得到的结果再入队.

代码实现

java 复制代码
class Solution {
    public int lastStoneWeight(int[] stones) {
        PriorityQueue<Integer> heap = new PriorityQueue<>((a,b)->b-a);
        //入队
        for (int stone : stones) 
            heap.offer(stone);
            
        while(heap.size()>1){
            int s1 = heap.poll();
            int s2 = heap.poll();
            if(s1!=s2)
                heap.offer(s1-s2);
        }
        return heap.isEmpty()?0:heap.poll();
    }
}

2. 数据流中的第K大元素(LC703)

数据流中的第K大元素

题目描述

解题思路

  1. 创建大小为K的小根堆
  2. 循环,元素依次进堆,判断堆的大小是否超过k,如果超过则令堆顶元素出堆。

代码实现

java 复制代码
class KthLargest {
    int k;
    int[] nums;
    PriorityQueue<Integer> heap = new PriorityQueue<>();
    public KthLargest(int k, int[] nums) {
        this.k = k;
        this.nums = nums;
        for(int x:nums){
            heap.offer(x);
            if(heap.size()>k)
                heap.poll();
        }
    }
    
    public int add(int val) {
        heap.offer(val);
        if(heap.size()>k)
            heap.poll();
        return heap.peek();
    }
}

3. 前K个高频单词(LC692)

前K个高频单词

题目描述

解题思路

利用堆来解决。

  1. 利用哈希表统计单词对应的次数
  2. 创建大小为k的堆
    • 对于频次,应该创建小根堆
    • 对于字典序(频次相同时),应该创建大根堆,让序列靠后的单词先出堆。
  3. 循环,让元素依次进堆,判断
  4. 提取结果,小根堆依次提取的结果是从小到大的,可以逆序存放在结果数组中,也可以调用reverse方法

代码实现

java 复制代码
class Solution {
    public List<String> topKFrequent(String[] words, int k) {
        List<String> ret = new ArrayList<>();
        HashMap<String,Integer> hash = new HashMap<>();
        //预处理,统计频次
        for(String word:words){
            if(hash.containsKey(word))
                hash.put(word,hash.get(word)+1);
            else
                hash.put(word,1);
        }

        //创建小根堆
        PriorityQueue<Map.Entry<String,Integer>> heap = new PriorityQueue<>((o1,o2) ->{
            if(o1.getValue().equals(o2.getValue()))
                    return o2.getKey().compareTo(o1.getKey());
                else
                    return o1.getValue() - o2.getValue();
        });

        //循环入队
        for(Map.Entry<String,Integer> entry:hash.entrySet()){
            heap.offer(entry);
            if(heap.size()>k)
                heap.poll();
        }

        //提取结果
        while(!heap.isEmpty())
            ret.add(heap.poll().getKey());
        
        Collections.reverse(ret);
        return ret;
    }
}

4. 数据流中的中位数(LC295)

数据流中的中位数

题目描述

解题思路

  • 利用大小堆来解决,创建大根堆存放左半边的元素,数量为m,小根堆存放右半边的元素,数量为n,保持m==nm==n+1m==n时,分别取两个堆的堆顶元素,求平均值;m==n+1 时,取大根堆的堆顶元素就是中位数。
  • addNum : 分类讨论,要保证m==nm==n+1
    • m==n时,取大根堆堆顶元素x
      • num<x 或 大根堆为空时,x添加到大根堆
      • 否则 添加到小根堆中,但是此时右边比左边数量多,取出堆顶元素y,放入小根堆中
    • m==n+1
      • num>x,直接放入右边
      • 否则先把x放入左边,x出堆放入右边。

代码实现

java 复制代码
class MedianFinder {
    PriorityQueue<Integer> left = new PriorityQueue<>((a,b)->b-a);
    PriorityQueue<Integer> right = new PriorityQueue<>();

    public MedianFinder() {
        
    }
     
    public void addNum(int num) {
        
        if(left.size()==right.size()){
            if(left.isEmpty() || left.peek()>=num)
                left.offer(num);
            else{
                right.offer(num);
                left.offer(right.poll());
            }
        }else{
            if(left.peek()<=num)
                right.offer(num);
            else{
                left.offer(num);
                right.offer(left.poll());
            }
        }
    }
    
    public double findMedian() {
        if(left.size()>right.size())
            return left.peek();
        else
            return (left.peek()+right.peek())/2.0 ;
    }
}
相关推荐
智算菩萨2 小时前
音频处理基础理论:从物理声波到数字信号完整知识体系
算法·机器学习·电脑·音视频
小高求学之路2 小时前
计算机视觉、YOLO算法模型训练、无人机监测人员密集自动识别
算法·yolo·计算机视觉
散峰而望3 小时前
【基础算法】剪枝与记忆化搜索:算法优化的双刃剑,效率倍增的实战指南
算法·机器学习·剪枝
m0_748873553 小时前
C++与Rust交互编程
开发语言·c++·算法
2401_8914821710 小时前
多平台UI框架C++开发
开发语言·c++·算法
88号技师10 小时前
2026年3月中科院一区SCI-贝塞尔曲线优化算法Bezier curve-based optimization-附Matlab免费代码
开发语言·算法·matlab·优化算法
t1987512810 小时前
三维点云最小二乘拟合MATLAB程序
开发语言·算法·matlab
无敌昊哥战神10 小时前
【LeetCode 257】二叉树的所有路径(回溯法/深度优先遍历)- Python/C/C++详细题解
c语言·c++·python·leetcode·深度优先
㓗冽11 小时前
8皇后·改-进阶题16
数据结构