力扣295. 数据流的中位数

优先队列

  • 思路:
    • 中位数是排序中间的数值:S1.M.S2
    • 可以使用两个优先队列来存放两边的数值,总是使得左侧的堆顶是最大的,右侧的堆顶是最小的,即使用大顶堆存放 S1,使用小顶堆存放S2,使得两个队列的 size 维持"平衡",则中位数就会在两个堆顶"附近"了;
    • 维持两个队列 size 平衡:
      • 数据先 push 的大顶堆,如果是 > M 的数,则会在堆顶;如果是 < M 的数,则会沉入队列中;
      • 然后将堆顶的数 push 到小顶堆,如果是 > M 的数,会沉入队列;如果是 < M 的数,会在堆顶;
      • 将大顶堆的堆顶 pop;(因为已经 push 到小顶堆)
      • 判断一下两个队列的size,如果大顶堆的 size 少了,将小顶堆的堆顶"漏"到大顶堆;(可以将两个队列组合成漏斗,更直观)
    • 此时的中位数:
      • 如果大顶堆 size 多,则中位数是其堆顶;
      • 否则,为两个堆顶的均值;
cpp 复制代码
class MedianFinder {
public:
    MedianFinder() {

    }
    
    void addNum(int num) {
        low.push(num);
        high.push(low.top());
        low.pop();

        if (low.size() < high.size()) {
            low.push(high.top());
            high.pop();
        }
    }
    
    double findMedian() {
        if (low.size() > high.size()) {
            return low.top();
        }

        return (low.top() + high.top()) / 2.0;
    }

private:
    std::priority_queue<int, std::vector<int>, std::less<int>> low;
    std::priority_queue<int, std::vector<int>, std::greater<int>> high;
};

/**
 * Your MedianFinder object will be instantiated and called as such:
 * MedianFinder* obj = new MedianFinder();
 * obj->addNum(num);
 * double param_2 = obj->findMedian();
 */
相关推荐
lee_curry几秒前
第三章 jvm中的对象和执行引擎
java·jvm·执行引擎
哆啦刘小洋1 分钟前
【LeetCode每日一题】:2033(贪心+快速排序魔改)
算法·leetcode
wang09071 分钟前
Linux性能优化之文件系统基础介绍
java·linux·性能优化
WolfGang0073214 分钟前
代码随想录算法训练营 Day48 | 图论 part06
算法·图论
cheems95279 分钟前
[算法手记] 动态规划 ,二维费用限制背包问题如何处理
算法·动态规划
迷藏49410 分钟前
# 发散创新:用Locust实现高并发场景下的精准压力测试与性能调优实战在现代微服务架构中,**接口稳定性与响应速度**已成为衡量
java·python·微服务·架构·压力测试
空中海15 分钟前
Nacos 2: Spring Boot Demo 实战
java·spring boot·后端
Chase_______15 分钟前
LeetCode 1343 题解:定长滑动窗口经典入门题,从暴力枚举到高效优化一文搞懂
算法·leetcode·职场和发展
样例过了就是过了16 分钟前
LeetCode热题100 单词拆分
c++·算法·leetcode·动态规划·哈希算法
土豆.exe20 分钟前
Cast Attack:Java 中 Ghost Bits(幽灵比特)引发的新型安全威胁——Java 生态里被忽视的底层风险引发一系列绕过
java·python·安全