-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathsparse_table_rsq.cpp
executable file
·64 lines (48 loc) · 1.32 KB
/
sparse_table_rsq.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
template <class T, class F = function <T(const T &, const T &)> >
class SparseTable {
int _N;
int _LOG;
vector <T> _A;
vector < vector <T> > ST;
vector <int> Log;
F func;
public :
SparseTable() = default;
template <class iter>
SparseTable(iter _begin, iter _end, F _func = [](T a, T b) { return a + b; }) : func(_func) {
_N = distance(_begin, _end);
Log.assign(_N + 1, 0);
for(int i = 2; i <= _N; ++i)
Log[i] = Log[i >> 1] + 1;
_LOG = Log[_N];
_A.assign(_N + 1, 0);
ST.assign(_N + 1, vector <T> (_LOG + 1, 0));
__typeof(_begin) i = _begin;
for(int j = 1; i != _end; ++i, ++j)
_A[j] = *i;
build();
}
void build() {
for(int i = 1; i <= _N; ++i)
ST[i][0] = _A[i];
for(int j = 1, k, d; j <= _LOG; ++j) {
k = (1 << j);
d = (k >> 1);
for(int i = 1; i + k - 1 <= _N; ++i) {
T const & x = ST[i][j - 1]; // starting subarray at index = i with length = 2^{j - 1}
T const & y = ST[i + d][j - 1]; // starting subarray at index = i + d with length = 2^{j - 1}
ST[i][j] = func(x, y);
}
}
}
T query(int l, int r) {
int d = r - l + 1;
T ret = 0;
for(int i = l; d; i += lastBit(d), d -= lastBit(d))
ret = func(ret, ST[i][Log[lastBit(d)]]);
return ret;
}
int lastBit(int a) {
return (a & -a);
}
};