forked from fishercoder1534/Leetcode
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy path_295.java
110 lines (93 loc) · 3.68 KB
/
_295.java
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
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
package com.fishercoder.solutions;
import java.util.Collections;
import java.util.PriorityQueue;
import java.util.Queue;
/**
* 295. Find Median from Data Stream
*
* Median is the middle value in an ordered integer list.
* If the size of the list is even, there is no middle value.
* So the median is the mean of the two middle value.
Examples:
[2,3,4] , the median is 3
[2,3], the median is (2 + 3) / 2 = 2.5
Design a data structure that supports the following two operations:
void addNum(int num) - Add a integer number from the data stream to the data structure.
double findMedian() - Return the median of all elements so far.
For example:
addNum(1)
addNum(2)
findMedian() -> 1.5
addNum(3)
findMedian() -> 2
*/
public class _295 {
/**A few key points for both following solutions:
*
* 1. always keep one queue one element more than the other if the number is odd, offer into that one
* first, then poll from that queue and offer into the other queue, then check whether that queue is smaller
* in size than the other, if so, poll one from the other queue and offer it into this queue
*
* 2. only need to check whether this bigger queue size is greater than the other queue when returning.*/
public static class Solution1 {
public static class MedianFinder {
private Queue<Long> large;
private Queue<Long> small;
public MedianFinder() {
large = new PriorityQueue<>();
small = new PriorityQueue<>(Collections.reverseOrder());
}
// Adds a number into the data structure.
public void addNum(int num) {
large.offer((long) num);
small.offer(large.poll());
if (large.size() < small.size()) {
large.offer(small.poll());
}
}
// Returns the median of current data stream
public double findMedian() {
if (large.size() > small.size()) {
return large.peek();
}
return (large.peek() + small.peek()) / 2.0;
}
}
}
public static class Solution2 {
public static class MedianFinder {
/**
* credit: https://discuss.leetcode.com/topic/27521/short-simple-java-c-python-o-log-n-o-1
* The idea is for sure to use two heaps, one is max heap, one is min heap, we always let the max heap be one element
* bigger than min heap if the total number of elements is not even.
* we could always get the median in O(1) time.
* 1. use Long type to avoid overflow
* 2. negate the numbers for small heap to save the effort for writing a reverse comparator, brilliant!
*/
private Queue<Long> large;
private Queue<Long> small;
/**
* initialize your data structure here.
*/
public MedianFinder() {
large = new PriorityQueue<>();
small = new PriorityQueue<>();
}
// Adds a number into the data structure.
public void addNum(int num) {
large.offer((long) num);
small.offer(-large.poll());
if (large.size() < small.size()) {
large.offer(-small.poll());
}
}
// Returns the median of current data stream
public double findMedian() {
if (large.size() > small.size()) {
return large.peek();
}
return (large.peek() - small.peek()) / 2.0;
}
}
}
}