每日算法:数据流的中位数

开发 前端 算法
如果插入元素比大顶堆的堆顶要大,则将该元素插入到小顶堆中;如果要小,则插入到大顶堆中。

[[431427]]

本文转载自微信公众号「三分钟学前端」,作者sisterAn  。转载本文请联系三分钟学前端公众号。

中位数是有序列表中间的数。如果列表长度是偶数,中位数则是中间两个数的平均值。

例如,

[2,3,4] 的中位数是 3

[2,3] 的中位数是 (2 + 3) / 2 = 2.5

设计一个支持以下两种操作的数据结构:

  • void addNum(int num)- 从数据流中添加一个整数到数据结构中。
  • double findMedian() - 返回目前所有元素的中位数。

示例:

  1. addNum(1) 
  2. addNum(2) 
  3. findMedian() -> 1.5 
  4. addNum(3)  
  5. findMedian() -> 2 

进阶:

  • 如果数据流中所有整数都在 0 到 100 范围内,你将如何优化你的算法?
  • 如果数据流中 99% 的整数都在 0 到 100 范围内,你将如何优化你的算法?

看到这个动态数组获取中位数问题,不要太激动,这太适合使用堆了,考察的就是堆的经典应用:中位数问题,详情可查看 前端进阶算法9:看完这篇,再也不怕堆排序、Top K、中位数问题面试了

解法:利用堆

解题思路:

这里需要维护两个堆:

  • 大顶堆:用来存取前 n/2 个小元素,如果 n 为奇数,则用来存取前 Math.floor(n/2) + 1 个元素
  • 小顶堆:用来存取后 n/2 个小元素

那么,根据题目要求,中位数就为:

  • n 为奇数:中位数是大顶堆的堆顶元素
  • n 为偶数:中位数是大顶堆的堆顶元素与小顶堆的堆顶元素的平均值

当数组为动态数组时,每当数组中插入一个元素时,都需要如何调整堆喃?

如果插入元素比大顶堆的堆顶要大,则将该元素插入到小顶堆中;如果要小,则插入到大顶堆中。

当插入完成后,如果大顶堆、小顶堆中元素的个数不满足我们已上的要求,我们就需要不断的将大顶堆的堆顶元素或小顶堆的堆顶元素移动到另一个堆中,直到满足要求

代码实现:

  1. let MedianFinder = function() { 
  2.     // 大顶堆,用来保存前 n/2 小的元素 
  3.     this.lowHeap = new MaxHeap() 
  4.     // 小顶堆,用来保存后 n/2 小的元素 
  5.     this.hightHeap = new MinHeap() 
  6. }; 
  7. // 插入元素 
  8. MedianFinder.prototype.addNum = function(num) { 
  9.     // 如果大顶堆为空或大顶堆堆顶元素小于num,则插入大顶堆 
  10.     // 否则插入到小顶堆中 
  11.     if(!this.lowHeap.getSize() || num < this.lowHeap.getHead()) { 
  12.         // 比大顶堆的堆顶小,插入到大顶堆中 
  13.         this.lowHeap.insert(num) 
  14.     } else { 
  15.         // 比小顶堆的堆顶大,插入到小顶堆中 
  16.         this.hightHeap.insert(num) 
  17.     } 
  18.  
  19.     // 比较大小顶堆的是否依然保持平衡 
  20.     if(this.lowHeap.getSize() - this.hightHeap.getSize() > 1) { 
  21.         // 大顶堆往小顶堆迁移 
  22.         this.hightHeap.insert(this.lowHeap.removeHead()) 
  23.     } 
  24.     if(this.hightHeap.getSize() > this.lowHeap.getSize()) { 
  25.         // 小顶堆向大顶堆迁移 
  26.         this.lowHeap.insert(this.hightHeap.removeHead()) 
  27.     } 
  28. }; 
  29. // 获取中位数 
  30. MedianFinder.prototype.findMedian = function() { 
  31.     if(this.lowHeap.getSize() && this.lowHeap.getSize() === this.hightHeap.getSize()) { 
  32.         return (this.lowHeap.getHead() + this.hightHeap.getHead())/2 
  33.     } 
  34.     return this.lowHeap.getHead() 
  35. }; 

其中小顶堆定义:

  1. // 小顶堆 
  2. let MinHeap = function() { 
  3.     let heap = [,] 
  4.     // 堆中元素数量 
  5.     this.getSize = ()=> heap.length - 1 
  6.     // 插入 
  7.     this.insert = (key) => { 
  8.         heap.push(key
  9.         // 获取存储位置 
  10.         let i = heap.length-1 
  11.         while (Math.floor(i/2) > 0 && heap[i] < heap[Math.floor(i/2)]) {   
  12.             swap(heap, i, Math.floor(i/2)); // 交换  
  13.             i = Math.floor(i/2);  
  14.         } 
  15.     } 
  16.     // 删除堆头并返回 
  17.     this.removeHead = () => { 
  18.         if(heap.length > 1) { 
  19.             if(heap.length === 2) return heap.pop() 
  20.             let num = heap[1] 
  21.             heap[1] = heap.pop() 
  22.             heapify(1) 
  23.             return num 
  24.         } 
  25.         return null 
  26.     } 
  27.     // 获取堆头 
  28.     this.getHead = () => { 
  29.         return heap.length > 1 ? heap[1]:null 
  30.     } 
  31.     // 堆化 
  32.     let heapify = (i) => { 
  33.         let k = heap.length-1 
  34.         // 自上而下式堆化 
  35.         while(true) { 
  36.             let minIndex = i 
  37.             if(2*i <= k && heap[2*i] < heap[i]) { 
  38.                 minIndex = 2*i 
  39.             } 
  40.             if(2*i+1 <= k && heap[2*i+1] < heap[minIndex]) { 
  41.                 minIndex = 2*i+1 
  42.             } 
  43.             if(minIndex !== i) { 
  44.                 swap(heap, i, minIndex) 
  45.                 i = minIndex 
  46.             } else { 
  47.                 break 
  48.             } 
  49.         } 
  50.     }  
  51.     let swap = (arr, i, j) => { 
  52.         let temp = arr[i] 
  53.         arr[i] = arr[j] 
  54.         arr[j] = temp 
  55.     } 

大顶堆定义:

  1. // 大顶堆 
  2. let MaxHeap = function() { 
  3.     let heap = [,] 
  4.     // 堆中元素数量 
  5.     this.getSize = ()=>heap.length - 1 
  6.     // 插入大顶堆 
  7.     this.insert = (key) => { 
  8.         heap.push(key
  9.         // 获取存储位置 
  10.         let i = heap.length-1 
  11.         while (Math.floor(i/2) > 0 && heap[i] > heap[Math.floor(i/2)]) {   
  12.             swap(heap, i, Math.floor(i/2)); // 交换  
  13.             i = Math.floor(i/2);  
  14.         } 
  15.     } 
  16.     // 获取堆头 
  17.     this.getHead = () => { 
  18.         return heap.length > 1 ? heap[1]:null 
  19.     } 
  20.     // 删除堆头并返回 
  21.     this.removeHead = () => { 
  22.         if(heap.length > 1) { 
  23.             if(heap.length === 2) return heap.pop() 
  24.             let num = heap[1] 
  25.             heap[1] = heap.pop() 
  26.             heapify(1) 
  27.             return num 
  28.         } 
  29.         return null 
  30.     } 
  31.     // 堆化 
  32.     let heapify = (i) => { 
  33.         let k = heap.length-1 
  34.         // 自上而下式堆化 
  35.         while(true) { 
  36.             let maxIndex = i 
  37.             if(2*i <= k && heap[2*i] > heap[i]) { 
  38.                 maxIndex = 2*i 
  39.             } 
  40.             if(2*i+1 <= k && heap[2*i+1] > heap[maxIndex]) { 
  41.                 maxIndex = 2*i+1 
  42.             } 
  43.             if(maxIndex !== i) { 
  44.                 swap(heap, i, maxIndex) 
  45.                 i = maxIndex 
  46.             } else { 
  47.                 break 
  48.             } 
  49.         } 
  50.     }  
  51.     let swap = (arr, i, j) => { 
  52.         let temp = arr[i] 
  53.         arr[i] = arr[j] 
  54.         arr[j] = temp 
  55.     } 

复杂度分析:

时间复杂度:由于插入元素到堆的时间复杂度为 O(logn),为树的高度;移动堆顶元素都需要堆化,时间复杂度也为O(logn);所以,插入( addNum )的时间复杂度为 O(logn) ,每次插入完成后求中位数仅仅需要返回堆顶元素即可, findMedian 时间复杂度为 O(1)

空间复杂度:O(n)

如果数据流中所有整数都在 0 到 100 范围内,我们可以尝试使用计数排序,但计数排序的时间复杂度是O(n + m),其中 m 表示数据范围,复杂度较高,这里不适合,计数排序比较适合静态数组前k个最值问题 leetcode347:前 K 个高频元素

leetcode:https://leetcode-cn.com/problems/find-median-from-data-stream/solution/javascriptshu-ju-liu-de-zhong-wei-shu-by-user7746o/

 

责任编辑:武晓燕 来源: 三分钟学前端
相关推荐

2021-06-29 19:24:42

数据流数据排序

2017-11-16 19:26:34

海量数据算法计算机

2011-12-14 15:57:13

javanio

2016-11-14 19:01:36

数据流聊天系统web

2022-03-18 08:57:17

前端数据流选型

2009-08-19 10:41:12

Java输入数据流

2011-04-14 14:43:38

SSISTransformat

2019-12-19 14:38:08

Flink SQL数据流Join

2012-07-30 08:31:08

Storm数据流

2011-04-19 09:18:02

SSIS数据转换

2014-02-11 08:51:15

亚马逊PaaSAppStream

2013-10-21 10:58:50

微软大数据SQL Server

2009-07-15 09:06:11

Linux图形系统X11的CS架构

2020-02-06 19:12:36

Java函数式编程编程语言

2014-12-02 10:56:47

TCPIP交互数据流

2020-08-20 11:24:31

物联网数据技术

2023-08-31 16:47:05

反应式编程数据流

2024-04-18 09:02:11

数据流Mixtral混合模型

2023-03-17 07:39:54

开源数据流技术

2013-10-12 12:56:46

点赞
收藏

51CTO技术栈公众号