力扣295. 数据流的中位数(java,堆解法)

Problem: 295. 数据流的中位数

文章目录

题目描述


思路

由于该题目的数据是动态的 我们可以维护两个来解决该问题

1.维护一个大顶堆,一个小顶堆

2.每个堆中元素个数接近n/2;如果n是偶数,两个堆中的数据个数都是n/2;如果n是奇数,则大顶堆中有n/2 + 1个数据,小顶堆中有n/2个数据

3.大顶堆中的数据值都要小于小顶堆中的数据值

即大顶堆中的堆顶元素就是中位数

解题方法

1.(创建堆)按思路创建一个大顶堆和小顶堆

2.(维护堆):

2.1.如果新插入数据小于等于大顶堆,则将其插入到大顶堆中,否则插入到小顶堆;

2.2.插入数据后,两个堆中的数据量个数不满足思路中的要求2,则我们需要从一个堆中不停的将堆顶元素移动到另一个堆

复杂度

时间复杂度:

a d d N u m : O ( l o g n ) addNum:O(logn) addNum:O(logn)
f i n d M e d i a n : O ( 1 ) findMedian:O(1) findMedian:O(1)

空间复杂度:

O ( n ) O(n) O(n)

Code

java 复制代码
class MedianFinder {
    /*维护一个大顶堆和小顶堆*/
    private PriorityQueue<Integer> minQueue = new PriorityQueue<>(new Comparator<Integer>() {
        @Override
        public int compare(Integer o1, Integer o2) {
            return o1 - o2;
        }
    });
    private PriorityQueue<Integer> maxQueue = new PriorityQueue<>(new Comparator<Integer>() {
        @Override
        public int compare(Integer o1, Integer o2) {
            return o2 - o1;
        }
    });

    public MedianFinder() {

    }

    /**
     * 数据流插入数据
     *
     * @param num 待插入的数据
     */
    public void addNum(int num) {
        //如果插入数据小于等于大顶堆堆顶元素,大顶堆直接插入
        if (maxQueue.isEmpty() || num <= maxQueue.peek()) {
            maxQueue.add(num);
        } else {
            minQueue.add(num);
        }
        //大顶堆数据量不能小于小顶堆
        while (maxQueue.size() < minQueue.size()) {
            Integer minQueueElement = minQueue.poll();
            maxQueue.add(minQueueElement);
        }
        //小顶堆数据量可以比大顶堆小一个
        while (minQueue.size() < maxQueue.size() - 1) {
            Integer maxQueueElement = maxQueue.poll();
            minQueue.add(maxQueueElement);
        }
    }

    /**
     * 找出中位数
     *
     * @return double
     */
    public double findMedian() {
        //如果大顶堆数据量大于小顶堆
        if (maxQueue.size() > minQueue.size()) {
            return maxQueue.peek();
        } else {
            return (maxQueue.peek() + minQueue.peek()) / 2f;
        }
    }
}


/**
 * Your MedianFinder object will be instantiated and called as such:
 * MedianFinder obj = new MedianFinder();
 * obj.addNum(num);
 * double param_2 = obj.findMedian();
 */
相关推荐
夫唯不争,故无尤也1 分钟前
HTTP方法详解:GET、POST、PUT、DELETE
开发语言·windows·python
zh路西法9 分钟前
【宇树机器人强化学习】(一):PPO算法的python实现与解析
python·深度学习·算法·机器学习·机器人
小钻风33669 分钟前
Optional:告别NullPointerException的优雅方案
开发语言·python
章小幽21 分钟前
LeetCode-35.搜索插入位置
数据结构·算法·leetcode
科技块儿23 分钟前
多语言技术栈如何共用IP离线库?Java、Python、Go 的加载实践
java·python·tcp/ip
fawubio_A33 分钟前
毕业设计 深度学习卷积神经网络垃圾分类系统
python·cnn·毕业设计·毕设
chools37 分钟前
一篇文章带你搞懂Java“设计模式”! - - 超长文(涵盖23种)万字总结!【汇总篇】
java·开发语言·设计模式
良逍Ai出海38 分钟前
OpenClaw 新手最该先搞懂的 2 套命令
android·java·数据库
6+h1 小时前
【Spring】深度剖析IoC
java·后端·spring
与虾牵手1 小时前
大模型流式输出 Streaming API 完整教程:从原理到踩坑,一篇搞定
python·aigc·ai编程