数据流的中位数

题目链接

295. 数据流的中位数 - 力扣(LeetCode)

思路

利用数据结构的堆来解决这个问题 ------ 大根堆 小根堆

大根堆里面的内容 特征是 父节点的值 大于或者等于 其子节点的值

小根堆里面的内容 特征是 父节点的值 小于或者等于 其子节点的值

大根堆里面数据 其实就是 其中位数的 左边 的 ------ 比中位数小的内容

小根堆里面数据 其实就是 其中位数的 右边 的 ------ 比中位数大的内容

我们维护这俩个堆,大根堆可以比小根堆多一个,此时 数据是 奇数 中位数 是大根堆的堆顶,当大根堆和小根堆数据量一致时,代表此时数据是 偶数,返回 大根堆的堆顶 和 小根堆 的堆顶 俩个数字除以2 就是最后的内容

c++ java python 都有现成的数据结构可以使用,

js需要手写(参考了一个题解的写法)

代码:

javascript 复制代码
var MedianFinder = function() {
    function createHeap(type) { // type: 0-最小堆, 1-最大堆
        const h = [null]; // 索引从1开始,方便计算
        let size = 0;

        function push(val) {
            h[++size] = val;
            up(size);
        }

        function up(u) {
            // 当还有父节点且需要交换时
            while (u >> 1 >= 1) {
                const parent = u >> 1;
                // 最小堆:子节点 < 父节点 时交换
                // 最大堆:子节点 > 父节点 时交换
                if ((type === 0 && h[u] < h[parent]) || 
                    (type === 1 && h[u] > h[parent])) {
                    [h[u], h[parent]] = [h[parent], h[u]];
                    u = parent;
                } else {
                    break;
                }
            }
        }

        function down(u) {
            while (u * 2 <= size) {
                let t = u;
                const left = u * 2;
                const right = u * 2 + 1;
                
                // 和左子节点比较
                if ((type === 0 && h[t] > h[left]) || 
                    (type === 1 && h[t] < h[left])) {
                    t = left;
                }
                // 和右子节点比较
                if (right <= size && (
                    (type === 0 && h[t] > h[right]) || 
                    (type === 1 && h[t] < h[right]))) {
                    t = right;
                }
                
                if (t !== u) {
                    [h[u], h[t]] = [h[t], h[u]];
                    u = t;
                } else {
                    break;
                }
            }
        }

        function pop() {
            if (size === 0) return null;
            const top = h[1];
            h[1] = h[size--];
            down(1);
            return top;
        }

        function peek() {
            return h[1];
        }

        function getSize() {
            return size;
        }

        function isEmpty() {
            return size === 0;
        }

        return {
            push,
            pop,
            peek,
            getSize,
            isEmpty
        };
    }

    // 最大堆(存较小的一半)
    this.maxHeap = createHeap(1);
    // 最小堆(存较大的一半)
    this.minHeap = createHeap(0);
};

MedianFinder.prototype.addNum = function(num) {
    // 标准解法:先加入最大堆,然后把最大堆的最大值移到最小堆
    if (this.maxHeap.isEmpty() || num <= this.maxHeap.peek()) {
        this.maxHeap.push(num);
    } else {
        this.minHeap.push(num);
    }
    
    // 平衡两个堆的大小
    // 最大堆可以比最小堆多一个元素,但不能少
    if (this.maxHeap.getSize() > this.minHeap.getSize() + 1) {
        this.minHeap.push(this.maxHeap.pop());
    } else if (this.minHeap.getSize() > this.maxHeap.getSize()) {
        this.maxHeap.push(this.minHeap.pop());
    }
};

MedianFinder.prototype.findMedian = function() {
    if (this.maxHeap.getSize() > this.minHeap.getSize()) {
        // 奇数个元素,最大堆多一个
        return this.maxHeap.peek();
    } else {
        // 偶数个元素,取平均值
        return (this.maxHeap.peek() + this.minHeap.peek()) / 2;
    }
};
相关推荐
lichenyang45311 小时前
Docker 学习笔记(一):为什么需要镜像、容器和仓库?
前端
kyriewen12 小时前
别再对着 TypeScript 报错发呆了:我把 10 个最常见的红色波浪线翻译成了人话
前端·javascript·typescript
IT_陈寒12 小时前
SpringBoot自动配置的坑,我的API突然就404了
前端·人工智能·后端
free3512 小时前
从 0 实现一个 Tiny JavaScript VM:项目架构拆解
javascript
奇奇怪怪的12 小时前
Embedding 模型 10+ 横向评测
前端
陈广亮12 小时前
Monorepo 从 0 到 1 实操指南 2026 版:pnpm catalogs + Turborepo 2.x + changesets 全链路
前端
子兮曰13 小时前
OpenMontage 深度解剖:你的 AI 编程助手,其实是个视频工作室
前端·后端·ai编程
敲代码的鱼13 小时前
PDF 预览与签名批注写回 支持安卓 iOS 鸿蒙 UTS插件
android·前端·ios
子兮曰13 小时前
前端工具链的「Rust 化」:一场没有赢家的军备竞赛?
前端·后端·rust
Hyyy14 小时前
Function Calling / Tool Use的原理和实现模式
前端·llm·ai编程