-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathfind_median_from_data_stream.go
55 lines (47 loc) · 1.38 KB
/
find_median_from_data_stream.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
package main
import "container/heap"
type Heap struct {
Values []int
LessFunc func(int, int) bool
}
func (h *Heap) Less(i, j int) bool { return h.LessFunc(h.Values[i], h.Values[j]) }
func (h *Heap) Swap(i, j int) { h.Values[i], h.Values[j] = h.Values[j], h.Values[i] }
func (h *Heap) Len() int { return len(h.Values) }
func (h *Heap) Peek() int { return h.Values[0] }
func (h *Heap) Pop() (v interface{}) {
h.Values, v = h.Values[:h.Len()-1], h.Values[h.Len()-1]
return v
}
func (h *Heap) Push(v interface{}) { h.Values = append(h.Values, v.(int)) }
func NewHeap(less func(int, int) bool) *Heap {
return &Heap{LessFunc: less}
}
type MedianFinder struct {
smallHeap *Heap
largeHeap *Heap
}
func Constructor() MedianFinder {
return MedianFinder{
smallHeap: NewHeap(func(a, b int) bool {
return a > b
}),
largeHeap: NewHeap(func(a, b int) bool {
return a < b
}),
}
}
func (mf *MedianFinder) AddNum(num int) {
if (mf.smallHeap.Len()+mf.largeHeap.Len())%2 == 0 {
heap.Push(mf.largeHeap, num)
heap.Push(mf.smallHeap, heap.Pop(mf.largeHeap))
} else {
heap.Push(mf.smallHeap, num)
heap.Push(mf.largeHeap, heap.Pop(mf.smallHeap))
}
}
func (mf *MedianFinder) FindMedian() float64 {
if (mf.smallHeap.Len()+mf.largeHeap.Len())%2 == 0 {
return (float64(mf.smallHeap.Peek()) + float64(mf.largeHeap.Peek())) / 2
}
return float64(mf.smallHeap.Peek())
}