Kyopro Library
 
読み取り中…
検索中…
一致する文字列を見つけられません
segtree_beats.hpp
[詳解]
1#include "../../kyopro_library/template.hpp"
2
3/// @brief Segment Tree Beat!
4/// @ref https://nyaannyaan.github.io/library/segment-tree/segment-tree-beats.hpp
5struct AngelBeats {
6 struct alignas(32) Node {
7 ll sum = 0, g1 = 0, l1 = 0;
8 ll g2 = -INFL, gc = 1, l2 = INFL, lc = 1, add = 0;
9 };
11 ll n, log;
13 AngelBeats(int _n) : AngelBeats(vector<ll>(_n)) {}
14 AngelBeats(const vector<ll>& vc) {
15 n = 1, log = 0;
16 while (n < (int)vc.size()) n <<= 1, log++;
17 v.resize(2 * n);
18 for (ll i = 0; i < (int)vc.size(); ++i) {
19 v[i + n].sum = v[i + n].g1 = v[i + n].l1 = vc[i];
20 }
21 for (ll i = n - 1; i; --i) update(i);
22 }
23 /// @brief [l, r) に対し、v[i] <- min(v[i], x)
24 void range_chmin(int l, int r, ll x) { inner_apply<1>(l, r, x); }
25 /// @brief [l, r) に対し、v[i] <- max(v[i], x)
26 void range_chmax(int l, int r, ll x) { inner_apply<2>(l, r, x); }
27 /// @brief [l, r) に対し、v[i] <- v[i] + x
28 void range_add(int l, int r, ll x) { inner_apply<3>(l, r, x); }
29 /// @brief [l, r) に対し、v[i] <- x
30 void range_update(int l, int r, ll x) { inner_apply<4>(l, r, x); }
31 /// @brief min[l, r)(v[i]) を返す
32 ll range_min(int l, int r) { return inner_fold<1>(l, r); }
33 /// @brief max[l, r)(v[i]) を返す
34 ll range_max(int l, int r) { return inner_fold<2>(l, r); }
35 /// @brief sum[l, r)(v[i]) を返す
36 ll range_sum(int l, int r) { return inner_fold<3>(l, r); }
37
38private:
39 void update(int k) {
40 Node& p = v[k];
41 Node& l = v[k * 2 + 0];
42 Node& r = v[k * 2 + 1];
43 p.sum = l.sum + r.sum;
44 if (l.g1 == r.g1) {
45 p.g1 = l.g1;
46 p.g2 = max(l.g2, r.g2);
47 p.gc = l.gc + r.gc;
48 } else {
49 bool f = l.g1 > r.g1;
50 p.g1 = f ? l.g1 : r.g1;
51 p.gc = f ? l.gc : r.gc;
52 p.g2 = max(f ? r.g1 : l.g1, f ? l.g2 : r.g2);
53 }
54 if (l.l1 == r.l1) {
55 p.l1 = l.l1;
56 p.l2 = min(l.l2, r.l2);
57 p.lc = l.lc + r.lc;
58 } else {
59 bool f = l.l1 < r.l1;
60 p.l1 = f ? l.l1 : r.l1;
61 p.lc = f ? l.lc : r.lc;
62 p.l2 = min(f ? r.l1 : l.l1, f ? l.l2 : r.l2);
63 }
64 }
65 void push_add(int k, ll x) {
66 Node& p = v[k];
67 p.sum += x << (log + __builtin_clz(k) - 31);
68 p.g1 += x;
69 p.l1 += x;
70 if (p.g2 != -INFL) p.g2 += x;
71 if (p.l2 != INFL) p.l2 += x;
72 p.add += x;
73 }
74 void push_min(int k, ll x) {
75 Node& p = v[k];
76 p.sum += (x - p.g1) * p.gc;
77 if (p.l1 == p.g1) p.l1 = x;
78 if (p.l2 == p.g1) p.l2 = x;
79 p.g1 = x;
80 }
81 void push_max(int k, ll x) {
82 Node& p = v[k];
83 p.sum += (x - p.l1) * p.lc;
84 if (p.g1 == p.l1) p.g1 = x;
85 if (p.g2 == p.l1) p.g2 = x;
86 p.l1 = x;
87 }
88 void push(int k) {
89 Node& p = v[k];
90 if (p.add != 0) {
91 push_add(k * 2 + 0, p.add);
92 push_add(k * 2 + 1, p.add);
93 p.add = 0;
94 }
95 if (p.g1 < v[k * 2 + 0].g1) push_min(k * 2 + 0, p.g1);
96 if (p.l1 > v[k * 2 + 0].l1) push_max(k * 2 + 0, p.l1);
97 if (p.g1 < v[k * 2 + 1].g1) push_min(k * 2 + 1, p.g1);
98 if (p.l1 > v[k * 2 + 1].l1) push_max(k * 2 + 1, p.l1);
99 }
100 void subtree_chmin(int k, ll x) {
101 if (v[k].g1 <= x) return;
102 if (v[k].g2 < x) {
103 push_min(k, x);
104 return;
105 }
106 push(k);
107 subtree_chmin(k * 2 + 0, x);
108 subtree_chmin(k * 2 + 1, x);
109 update(k);
110 }
111 void subtree_chmax(int k, ll x) {
112 if (x <= v[k].l1) return;
113 if (x < v[k].l2) {
114 push_max(k, x);
115 return;
116 }
117 push(k);
118 subtree_chmax(k * 2 + 0, x);
119 subtree_chmax(k * 2 + 1, x);
120 update(k);
121 }
122 template <int cmd>
123 inline void _apply(int k, ll x) {
124 if constexpr (cmd == 1) subtree_chmin(k, x);
125 if constexpr (cmd == 2) subtree_chmax(k, x);
126 if constexpr (cmd == 3) push_add(k, x);
127 if constexpr (cmd == 4) subtree_chmin(k, x), subtree_chmax(k, x);
128 }
129 template <int cmd>
130 void inner_apply(int l, int r, ll x) {
131 if (l == r) return;
132 l += n, r += n;
133 for (int i = log; i >= 1; i--) {
134 if (((l >> i) << i) != l) push(l >> i);
135 if (((r >> i) << i) != r) push((r - 1) >> i);
136 }
137 {
138 int l2 = l, r2 = r;
139 while (l < r) {
140 if (l & 1) _apply<cmd>(l++, x);
141 if (r & 1) _apply<cmd>(--r, x);
142 l >>= 1;
143 r >>= 1;
144 }
145 l = l2;
146 r = r2;
147 }
148 for (int i = 1; i <= log; i++) {
149 if (((l >> i) << i) != l) update(l >> i);
150 if (((r >> i) << i) != r) update((r - 1) >> i);
151 }
152 }
153 template <int cmd>
154 inline ll e() {
155 if constexpr (cmd == 1) return INFL;
156 if constexpr (cmd == 2) return -INFL;
157 return 0;
158 }
159 template <int cmd>
160 inline void op(ll& a, const Node& b) {
161 if constexpr (cmd == 1) a = min(a, b.l1);
162 if constexpr (cmd == 2) a = max(a, b.g1);
163 if constexpr (cmd == 3) a += b.sum;
164 }
165 template <int cmd>
166 ll inner_fold(int l, int r) {
167 if (l == r) return e<cmd>();
168 l += n, r += n;
169 for (int i = log; i >= 1; i--) {
170 if (((l >> i) << i) != l) push(l >> i);
171 if (((r >> i) << i) != r) push((r - 1) >> i);
172 }
173 ll lx = e<cmd>(), rx = e<cmd>();
174 while (l < r) {
175 if (l & 1) op<cmd>(lx, v[l++]);
176 if (r & 1) op<cmd>(rx, v[--r]);
177 l >>= 1;
178 r >>= 1;
179 }
180 if constexpr (cmd == 1) lx = min(lx, rx);
181 if constexpr (cmd == 2) lx = max(lx, rx);
182 if constexpr (cmd == 3) lx += rx;
183 return lx;
184 }
185};
Segment Tree Beat! https://nyaannyaan.github.io/library/segment-tree/segment-tree-beats....
void range_update(int l, int r, ll x)
[l, r) に対し、v[i] <- x
AngelBeats(const vector< ll > &vc)
AngelBeats(int _n)
vector< Node > v
ll range_sum(int l, int r)
sum[l, r)(v[i]) を返す
ll range_min(int l, int r)
min[l, r)(v[i]) を返す
void range_chmin(int l, int r, ll x)
[l, r) に対し、v[i] <- min(v[i], x)
ll range_max(int l, int r)
max[l, r)(v[i]) を返す
void range_add(int l, int r, ll x)
[l, r) に対し、v[i] <- v[i] + x
void range_chmax(int l, int r, ll x)
[l, r) に対し、v[i] <- max(v[i], x)
const ll INFL
Definition template.hpp:13