-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path315. Count of Smaller Numbers After Self.cpp
102 lines (96 loc) · 2.56 KB
/
315. Count of Smaller Numbers After Self.cpp
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
const int range = 2e4+1;
const int offset = 1e4;
struct stree {
int tree[range*2];
stree(){
fill(tree,tree+2*range,0);
}
void update(int val){
val += range;
tree[val]++;
while(val != 0){
val >>=1;
tree[val]++;
}
}
int count(int val) {
int l = range;
int r = val+range;
int sum=0;
while(l<=r) {
if((l%2) == 1) {
sum += tree[l];
l++;
}
if( (r%2) == 0) {
sum += tree[r];
r--;
}
l >>= 1;
r >>= 1;
}
return sum;
}
};
class Solution {
public:
// TLE: 50/65 test cases. binary search
// distance has linear time complaxicity
vector<int> countSmaller1(vector<int>& nums) {
int n = nums.size();
vector<int> result(n,0);
multiset<int> s;
s.insert(nums.back());
for(int i=n-2;i>=0;--i) {
auto it = s.lower_bound(nums[i]);
result[i] = distance(s.begin(),it);
s.insert(nums[i]);
}
return result;
}
// segment tree: 284 ms
vector<int> countSmaller2(vector<int>& nums) {
stree st;
int n = nums.size();
vector<int> result(n,0);
st.update(nums.back()+offset);
for(int i=n-2;i>=0;--i) {
st.update(nums[i]+offset);
result[i] = st.count(nums[i]+offset -1);
}
return result;
}
// older BIT solution : 253 ms
// read this -https://leetcode.com/problems/count-of-smaller-numbers-after-self/discuss/138154/The-C%2B%2B-merge-sort-template-for-pairs-'i'-'j'-problem
// Least Significant Bit of i
// #define LSB(i) ((i) & -(i))
vector<int> countSmaller(vector<int>& nums) {
const int range = 2e5;
const int offset = 1e4 + 1;
vector<int> BIT(range,0);
int n = nums.size();
vector<int> result(n);
auto update = [&](int idx,int val){
while(idx<range)
{
BIT[idx] += val;
idx += idx & -idx;
}
};
auto sum = [&](int idx){
int sum=0;
while(idx>0)
{
sum +=BIT[idx];
idx -= idx & -idx;
}
return sum;
};
for(int i=n-1;i>=0;i--)
{
result[i]= ( sum(nums[i] + offset -1 ) );
update(nums[i] + offset ,1);
}
return result;
}
};