Leetcode 295. 数据流的中位数 堆

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


中位数查找思想:用两个堆维护数据(大顶堆存左半部分数据,小顶堆存右半部分数据),堆顶分别为左半最大值和右半最小值,可 O (1) 获取中位数。

解法1:官解

cpp 复制代码
class MedianFinder {
public:
    //大顶堆存左半部分数据, 堆顶为左半最大值
    priority_queue<int, vector<int> ,less<int>> quemin;
    // 小顶堆存右半部分数据, 堆顶为右半最小值
    priority_queue<int, vector<int> ,greater<int>> quemax;
    MedianFinder() {
        
    }
    // 整个序列长度偶数:quemin长度=quemax长度
    // 整个序列长度奇数:quemin长度=quemax长度+1
    void addNum(int num) {
        if(quemin.empty() || num<=quemin.top()){
            quemin.push(num);
            if(quemax.size()+1<quemin.size()){
                quemax.push(quemin.top());
                quemin.pop();
            }
        }
        else{
            quemax.push(num);
            if(quemax.size()>quemin.size()){
                quemin.push(quemax.top());
                quemax.pop();
            }
        }
    }
    
    double findMedian() {
        if(quemin.size() > quemax.size()) return quemin.top();
        return (quemin.top()+quemax.top()) / 2.0;
    }
};

/**
 * Your MedianFinder object will be instantiated and called as such:
 * MedianFinder* obj = new MedianFinder();
 * obj->addNum(num);
 * double param_2 = obj->findMedian();
 */

解法2:

参考: 如何自然引入大小堆?简洁写法!(Python/Java/C++/Go/JS/Rust)

  • 如果整个序列长度偶数:left长度 = right长度
  • 如果整个序列长度奇数:left长度 = right长度+1

加入数字的过程中, 要时时刻刻满足以上两个要求,就可以用 left 中的最大值以及 right 中的最小值计算中位数。

分类讨论:

  1. 如果当前 left 的大小和 right 的大小相等:
  • 如果添加的数字 num 比较大
  • 如果添加的数字 num 比较小
  1. 如果当前 left 比 right 多 1 个数:
  • 如果添加的数字 num 比较大
  • 如果添加的数字 num 比较小
cpp 复制代码
class MedianFinder {
public:
    //大顶堆存左半部分数据, 堆顶为左半最大值
    priority_queue<int, vector<int> ,less<int>> left;
    // 小顶堆存右半部分数据, 堆顶为右半最小值
    priority_queue<int, vector<int> ,greater<int>> right;
    MedianFinder() {
        
    }
    // 整个序列长度偶数:left长度=right长度
    // 整个序列长度奇数:left长度=right长度+1
    void addNum(int num) {
        if(left.size()==right.size()){
            right.push(num);
            left.push(right.top());
            right.pop();
        }
        else{
            left.push(num);
            right.push(left.top());
            left.pop();
        }
    }
    
    double findMedian() {
        if(left.size() > right.size()) return left.top();
        return (left.top()+right.top()) / 2.0;
    }
};

/**
 * Your MedianFinder object will be instantiated and called as such:
 * MedianFinder* obj = new MedianFinder();
 * obj->addNum(num);
 * double param_2 = obj->findMedian();
 */
相关推荐
superior tigre1 小时前
739 每日温度
算法·leetcode·职场和发展
6Hzlia1 小时前
【Hot 100 刷题计划】 LeetCode 15. 三数之和 | C++ 排序+双指针
c++·算法·leetcode
北顾笙9803 小时前
day37-数据结构力扣
数据结构·算法·leetcode
6Hzlia5 小时前
【Hot 100 刷题计划】 LeetCode 189. 轮转数组 | C++ 三次反转经典魔法 (O(1) 空间)
c++·算法·leetcode
m0_629494736 小时前
LeetCode 热题 100-----13.最大子数组和
数据结构·算法·leetcode
田梓燊6 小时前
力扣:94.二叉树的中序遍历
数据结构·算法·leetcode
khalil10206 小时前
代码随想录算法训练营Day-38动态规划06 | 322. 零钱兑换、279.完全平方数、139.单词拆分、多重背包、总结
数据结构·c++·算法·leetcode·动态规划
阿Y加油吧6 小时前
二刷 LeetCode:300. 最长递增子序列 & 152. 乘积最大子数组 复盘笔记
笔记·算法·leetcode
6Hzlia6 小时前
Hot 100 刷题计划】 LeetCode 146. LRU 缓存 | C++ 哈希表+双向链表
c++·leetcode·缓存
我不是懒洋洋6 小时前
【数据结构】二叉树OJ(单值二叉树、检查两棵树是否相同、对称二叉树、二叉树的前序遍历、另一颗树的子树)
c语言·数据结构·c++·经验分享·算法·leetcode·visual studio