-
Notifications
You must be signed in to change notification settings - Fork 46
/
Copy pathsegment_tree.cpp
130 lines (115 loc) · 3.24 KB
/
segment_tree.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
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
const int maxn = 50500;
const int inf = 1 << 29;
const int maxl = 16; // = ceil(log_2(maxn))
int N = 1 << maxl;
//============= customize here =================
struct Range {
int max_left;
int max_left1;
int max_right;
int max_right1;
int max_mid1;
int sum;
int lo, hi;
int count() const { return hi - lo + 1; }
};
Range NULL_RANGE = {0, -inf, 0, -inf, -inf, 0, inf, -inf};
Range operator+(const Range& l, const Range& r) {
return {
max(l.sum + r.max_left, l.max_left),
max(l.sum + ((l.count() > 0) ? r.max_left : r.max_left1),
l.max_left1),
max(r.sum + l.max_right, r.max_right),
max(r.sum + ((r.count() > 0) ? l.max_right : l.max_right1),
r.max_right1),
max( l.max_mid1,
max(r.max_mid1,
max(l.max_right1 + r.max_left,
l.max_right + r.max_left1))),
l.sum + r.sum,
min(l.lo, r.lo), max(l.hi, r.hi),
};
}
enum ActionType { ACTION_ID = 0, ACTION_SET };
struct Action {
ActionType type;
int val;
Range apply(const Range& r) {
if (type == ACTION_ID) return r;
int w = r.count();
return {
max(0, val * w),
max(val, val * w),
max(0, val * w),
max(val, val * w),
max(val, val * w),
val * w,
r.lo, r.hi,
};
}
};
Action NULL_ACTION = { ACTION_ID, 0 };
Action operator+(const Action& a, const Action& b) {
if (b.type == ACTION_SET) return b;
else return a;
}
//============= library code =================
int left(int i) { return 2 * i; }
int right(int i) { return 2 * i + 1; }
struct Node {
Range r;
Action action;
Range range() { return action.apply(r); }
} t[1 << (maxl + 1)];
void maintain(int i) {
t[i].r = t[left(i)].range() + t[right(i)].range();
t[i].action = NULL_ACTION;
}
void update(int x, const Action& a, int ql, int qr);
void push_down(int x) {
update(left(x), t[x].action, -inf, inf);
update(right(x), t[x].action, -inf, inf);
t[x].r = t[x].action.apply(t[x].r);
t[x].action = NULL_ACTION;
}
void update(int x, const Action& a, int ql, int qr) {
if (t[x].r.hi < ql || t[x].r.lo > qr) return;
if (ql <= t[x].r.lo && t[x].r.hi <= qr) {
t[x].action = t[x].action + a;
return;
}
push_down(x);
update(left(x), a, ql, qr);
update(right(x), a, ql, qr);
maintain(x);
}
Range query(int x, int ql, int qr) {
if (t[x].r.hi < ql || t[x].r.lo > qr) return NULL_RANGE;
if (ql <= t[x].r.lo && t[x].r.hi <= qr) return t[x].range();
push_down(x);
return query(left(x), ql, qr) + query(right(x), ql, qr);
}
// nodes reside in t[N..2N - 1], initialize those first
void init() {
rep(i,0,N) t[N + i].r.lo = t[N + i].r.hi = i;
for (int i = N - 1; i >= 1; --i)
maintain(i);
}
// ============= example ==============
void run() {
int n;
cin >> n;
N = 1; while (N < n) N <<= 1;
rep(i,0,n) {
int x; cin >> x;
t[N + i].r = NULL_RANGE;
t[N + i].action = { ACTION_SET, x };
}
rep(i,n,N) t[N + i].r = NULL_RANGE;
init();
// update
int lo = 2, hi = 5;
update(1, { ACTION_SET, 1 }, lo, hi);
// query
const Range& r = query(1, lo, hi);
}