力扣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();
 */
相关推荐
李拾叁的摸鱼日常4 分钟前
Spring 框架中 RequestContextHolder 深度解析
java·架构
C++业余爱好者13 分钟前
JVM优化入门指南:JVM垃圾收集器(GC)介绍
java·开发语言·jvm
dhdjjsjs16 分钟前
Day34 PythonStudy
python
Trouvaille ~19 分钟前
【Java篇】基石与蓝图::Object 类与抽象类的双重奏
java·开发语言·javase·抽象类·类与对象·基础入门·object类
小光学长23 分钟前
基于ssm的美妆产品推荐系统rah0h134(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
java·数据库·spring
Light6031 分钟前
破局“数据孤岛”:构建业务、财务、指标三位一体的智能数据模型
java·大数据·开发语言
中文很快乐32 分钟前
从零到一:用 SpringBoot 打造 RESTful API 实战指南
java·spring boot·后端·restful
一个java开发33 分钟前
Dask 配置文件加载机制说明
大数据·python
泉城老铁33 分钟前
springboot+redis 如何实现订单的过期
java·后端·架构