寒假后半段一直都在外出旅游。。颓了好久。。qaq
旅游期间写了几道树剖模板题,贴上来。。
BZOJ 1036 没啥好说的,裸题
1 #include <cstdio> 2 #include <algorithm> 3 4 #define LEFT (segt[cur].l) 5 #define RIGHT (segt[cur].r) 6 #define MID (segt[cur].mid) 7 #define SUM (segt[cur].Sum) 8 #define MAX (segt[cur].Max) 9 #define LCH (cur << 1) 10 #define RCH ((cur << 1) | 1) 11 12 using namespace std; 13 14 void Get_Val(int &Ret) 15 { 16 Ret = 0; 17 char ch; 18 bool neg(false); 19 while (ch = getchar(), (ch > '9' || ch < '0') && ch != '-') 20 ; 21 if (ch == '-') 22 { 23 neg = true; 24 while (ch = getchar(), ch > '9' || ch < '0') 25 ; 26 } 27 do 28 { 29 (Ret *= 10) += ch - '0'; 30 } 31 while (ch = getchar(), ch >= '0' && ch <= '9'); 32 Ret = (neg ? -Ret : Ret); 33 } 34 35 const int Max_N(30050); 36 const int root(1); 37 38 int N; 39 int W[Max_N]; 40 int Total, Head[Max_N], To[Max_N << 1], Next[Max_N << 1]; 41 int Father[Max_N], Deep[Max_N], Size[Max_N], Son[Max_N], Top[Max_N], ID[Max_N]; 42 int Tot, Number[Max_N]; 43 44 struct node 45 { 46 int l, r, mid; 47 int Sum, Max; 48 }; 49 50 struct Segment_Tree 51 { 52 node segt[Max_N << 2]; 53 void Make_Tree(const int&, const int&, const int&); 54 inline 55 void pushup(const int &cur) 56 { 57 MAX = max(segt[LCH].Max, segt[RCH].Max); 58 SUM = segt[LCH].Sum + segt[RCH].Sum; 59 } 60 void Change(const int&, const int&, const int&); 61 int RMQ_Max(const int&, const int&, const int&); 62 int Query_Sum(const int&, const int&, const int&); 63 }; 64 Segment_Tree Tree; 65 66 void Segment_Tree::Make_Tree(const int &cur, const int &l, const int &r) 67 { 68 LEFT = l, RIGHT = r, MID = l + ((r - l) >> 1), SUM = 0, MAX = 0; 69 if (l == r) 70 { 71 SUM = MAX = W[ID[l]]; 72 return; 73 } 74 Make_Tree(LCH, l, MID); 75 Make_Tree(RCH, MID + 1, r); 76 pushup(cur); 77 } 78 79 void Segment_Tree::Change(const int &cur, const int &i, const int &x) 80 { 81 if (LEFT == i && RIGHT == i) 82 { 83 SUM = MAX = x; 84 return; 85 } 86 if (i <= MID) 87 Change(LCH, i, x); 88 else 89 Change(RCH, i, x); 90 pushup(cur); 91 } 92 93 int Segment_Tree::RMQ_Max(const int &cur, const int &l, const int &r) 94 { 95 if (l == LEFT && r == RIGHT) 96 return MAX; 97 if (r <= MID) 98 return RMQ_Max(LCH, l, r); 99 else 100 if (l > MID) 101 return RMQ_Max(RCH, l, r); 102 else 103 return max(RMQ_Max(LCH, l, MID), RMQ_Max(RCH, MID + 1, r)); 104 } 105 106 int Segment_Tree::Query_Sum(const int &cur, const int &l, const int &r) 107 { 108 if (l == LEFT && r == RIGHT) 109 return SUM; 110 if (r <= MID) 111 return Query_Sum(LCH, l, r); 112 else 113 if (l > MID) 114 return Query_Sum(RCH, l, r); 115 else 116 return Query_Sum(LCH, l, MID) + Query_Sum(RCH, MID + 1, r); 117 } 118 119 inline 120 void Add_Edge(const int &s, const int &t) 121 { 122 ++Total; 123 To[Total] = t; 124 Next[Total] = Head[s], Head[s] = Total; 125 } 126 127 void dfs1(const int &u, const int &fa) 128 { 129 Size[u] = 1; 130 int v; 131 for (int i = Head[u];i;i = Next[i]) 132 { 133 v = To[i]; 134 if (v != fa) 135 { 136 Father[v] = u, Deep[v] = Deep[u] + 1; 137 dfs1(v, u); 138 Size[u] += Size[v]; 139 if (Size[v] > Size[Son[u]]) 140 Son[u] = v; 141 } 142 } 143 } 144 145 void dfs2(const int &u, const int &fa) 146 { 147 Number[u] = ++Tot, ID[Tot] = u; 148 int v; 149 if (v = Son[u]) 150 { 151 Top[v] = Top[u]; 152 dfs2(v, u); 153 } 154 for (int i = Head[u];i;i = Next[i]) 155 { 156 v = To[i]; 157 if (v != fa && v != Son[u]) 158 { 159 Top[v] = v; 160 dfs2(v, u); 161 } 162 } 163 } 164 165 void init() 166 { 167 int a, b; 168 Get_Val(N); 169 for (int i = 1;i != N;++i) 170 { 171 Get_Val(a), Get_Val(b); 172 Add_Edge(a, b), Add_Edge(b, a); 173 } 174 for (int i = 1;i <= N;++i) 175 Get_Val(W[i]); 176 Deep[root] = 1, dfs1(root, Father[root] = -1); 177 Top[root] = root, dfs2(root, -1); 178 } 179 180 int Query_Max(int u, int v) 181 { 182 int Max(-0X7F7F7F7F); 183 int r1(Top[u]), r2(Top[v]); 184 while (r1 != r2) 185 { 186 if (Deep[r1] < Deep[r2]) 187 swap(u, v), swap(r1, r2); 188 Max = max(Max, Tree.RMQ_Max(1, Number[r1], Number[u])); 189 u = Father[r1]; 190 r1 = Top[u]; 191 } 192 if (Deep[u] > Deep[v]) 193 swap(u, v); 194 return max(Max, Tree.RMQ_Max(1, Number[u], Number[v])); 195 } 196 197 int Query_Sum(int u, int v) 198 { 199 int Sum(0); 200 int r1(Top[u]), r2(Top[v]); 201 while (r1 != r2) 202 { 203 if (Deep[r1] < Deep[r2]) 204 swap(u, v), swap(r1, r2); 205 Sum += Tree.Query_Sum(1, Number[r1], Number[u]); 206 u = Father[r1]; 207 r1 = Top[u]; 208 } 209 if (Deep[u] > Deep[v]) 210 swap(u, v); 211 return Sum + Tree.Query_Sum(1, Number[u], Number[v]); 212 } 213 214 void work() 215 { 216 Tree.Make_Tree(1, 1, N); 217 int Q, u, v; 218 char op[10]; 219 Get_Val(Q); 220 while (Q--) 221 { 222 scanf("%s", op), Get_Val(u), Get_Val(v); 223 if (*op == 'C') 224 Tree.Change(1, Number[u], v); 225 else 226 if (op[1] == 'M') 227 printf("%d\n", Query_Max(u, v)); 228 else 229 printf("%d\n", Query_Sum(u, v)); 230 } 231 } 232 233 int main() 234 { 235 init(); 236 work(); 237 return 0; 238 }
BZOJ 2157 边权转点权。。点权表示的是这个点到它父亲的边的权值,根节点的点权设置为无意义的值,比如和是0,最大值是-INF,最小值是INF。树剖修改查询的时候,只需要最后一步提升时,将两个点的lca无视即可。具体看代码。。写了7kbqaq
1 #include <cstdio> 2 #include <algorithm> 3 #define LEFT (segt[cur].l) 4 #define RIGHT (segt[cur].r) 5 #define MID (segt[cur].mid) 6 #define SUM (segt[cur].Sum) 7 #define MAX (segt[cur].Max) 8 #define MIN (segt[cur].Min) 9 #define TAG (segt[cur].Tag) 10 #define LCH (cur << 1) 11 #define RCH ((cur << 1) | 1) 12 13 using namespace std; 14 15 void Get_Val(int &Ret) 16 { 17 Ret = 0; 18 char ch; 19 bool neg(false); 20 while (ch = getchar(), (ch > '9' || ch < '0') && ch != '-') 21 ; 22 if (ch == '-') 23 { 24 neg = true; 25 while (ch = getchar(), ch > '9' || ch < '0') 26 ; 27 } 28 do 29 { 30 (Ret *= 10) += ch -'0'; 31 } 32 while (ch = getchar(), ch >= '0' && ch <= '9'); 33 if (neg) 34 Ret = -Ret; 35 } 36 37 const int Max_N(20050); 38 const int root(1); 39 const int INF(0X3F3F3F3F); 40 41 struct edge 42 { 43 int u, v; 44 }; 45 edge Edges[Max_N]; 46 47 int N; 48 int Total, Head[Max_N], To[Max_N << 1], Next[Max_N << 1], Weight[Max_N << 1]; 49 int Father[Max_N], Cost[Max_N], Deep[Max_N], Size[Max_N], Son[Max_N], Top[Max_N]; 50 int Dfs_Clock, Pre[Max_N], ID[Max_N]; 51 52 inline 53 void Add_Edge(const int &s, const int &t, const int &w) 54 { 55 ++Total; 56 To[Total] = t, Weight[Total] = w; 57 Next[Total] = Head[s], Head[s] = Total; 58 } 59 60 struct node 61 { 62 int l, r, mid; 63 int Sum, Max, Min; 64 bool Tag; 65 }; 66 67 struct Segment_Tree 68 { 69 node segt[Max_N << 2]; 70 void build_tree(const int&, const int&, const int&); 71 inline 72 void pushup(const int &cur) 73 { 74 SUM = segt[LCH].Sum + segt[RCH].Sum; 75 MAX = max(segt[LCH].Max, segt[RCH].Max), MIN = min(segt[LCH].Min, segt[RCH].Min); 76 } 77 inline 78 void pushdown(const int &cur) 79 { 80 if (TAG) 81 { 82 segt[LCH].Sum = -segt[LCH].Sum; 83 segt[RCH].Sum = -segt[RCH].Sum; 84 swap((segt[LCH].Max = -segt[LCH].Max), (segt[LCH].Min = -segt[LCH].Min)); 85 swap((segt[RCH].Max = -segt[RCH].Max), (segt[RCH].Min = -segt[RCH].Min)); 86 segt[LCH].Tag = !segt[LCH].Tag, segt[RCH].Tag = !segt[RCH].Tag; 87 TAG = false; 88 } 89 } 90 void change(const int&, const int&, const int&); 91 void opposite(const int&, const int&, const int&); 92 int Sum(const int&, const int&, const int&); 93 int Min(const int&, const int&, const int&); 94 int Max(const int&, const int&, const int&); 95 }; 96 Segment_Tree Tree; 97 98 void Segment_Tree::build_tree(const int &cur, const int &l, const int &r) 99 { 100 LEFT = l, RIGHT = r, MID = l + ((r - l) >> 1), TAG = false; 101 if (l == r) 102 { 103 if (ID[l] == root) 104 MAX = -INF, MIN = INF, SUM = 0; 105 else 106 MAX = MIN = SUM = Cost[ID[l]]; 107 return; 108 } 109 build_tree(LCH, l, MID), build_tree(RCH, MID + 1, r); 110 pushup(cur); 111 } 112 113 void Segment_Tree::change(const int &cur, const int &i, const int &x) 114 { 115 if (LEFT == i && RIGHT == i) 116 { 117 SUM = MAX = MIN = x; 118 return; 119 } 120 pushdown(cur); 121 if (i <= MID) 122 change(LCH, i, x); 123 else 124 change(RCH, i, x); 125 pushup(cur); 126 } 127 128 void Segment_Tree::opposite(const int &cur, const int &l, const int &r) 129 { 130 if (LEFT == l && RIGHT == r) 131 { 132 SUM = -SUM; 133 swap((MAX = -MAX), (MIN = -MIN)); 134 TAG = !TAG; 135 return; 136 } 137 pushdown(cur); 138 if (r <= MID) 139 opposite(LCH, l, r); 140 else 141 if (l > MID) 142 opposite(RCH, l, r); 143 else 144 { 145 opposite(LCH, l, MID); 146 opposite(RCH, MID + 1, r); 147 } 148 pushup(cur); 149 } 150 151 int Segment_Tree::Sum(const int &cur, const int &l, const int &r) 152 { 153 if (LEFT == l && RIGHT == r) 154 return SUM; 155 pushdown(cur); 156 if (r <= MID) 157 return Sum(LCH, l, r); 158 else 159 if (l > MID) 160 return Sum(RCH, l, r); 161 else 162 return Sum(LCH, l, MID) + Sum(RCH, MID + 1, r); 163 } 164 165 int Segment_Tree::Max(const int &cur, const int &l, const int &r) 166 { 167 if (LEFT == l && RIGHT == r) 168 return MAX; 169 pushdown(cur); 170 if (r <= MID) 171 return Max(LCH, l, r); 172 else 173 if (l > MID) 174 return Max(RCH, l, r); 175 else 176 return max(Max(LCH, l, MID), Max(RCH, MID + 1, r)); 177 } 178 179 int Segment_Tree::Min(const int &cur, const int &l, const int &r) 180 { 181 if (LEFT == l && RIGHT == r) 182 return MIN; 183 pushdown(cur); 184 if (r <= MID) 185 return Min(LCH, l, r); 186 else 187 if (l > MID) 188 return Min(RCH, l, r); 189 else 190 return min(Min(LCH, l, MID), Min(RCH, MID + 1, r)); 191 } 192 193 void Opposite(int u, int v) 194 { 195 int r1(Top[u]), r2(Top[v]); 196 while (r1 != r2) 197 { 198 if (Deep[r1] < Deep[r2]) 199 swap(u, v), swap(r1, r2); 200 Tree.opposite(1, Pre[r1], Pre[u]); 201 u = Father[r1], r1 = Top[u]; 202 } 203 if (Deep[u] > Deep[v]) 204 swap(u, v); 205 if (Pre[u] + 1 <= Pre[v]) 206 Tree.opposite(1, Pre[u] + 1, Pre[v]); 207 } 208 209 int Query_Sum(int u, int v) 210 { 211 int Ans(0), r1(Top[u]), r2(Top[v]); 212 while (r1 != r2) 213 { 214 if (Deep[r1] < Deep[r2]) 215 swap(u, v), swap(r1, r2); 216 Ans += Tree.Sum(1, Pre[r1], Pre[u]); 217 u = Father[r1], r1 = Top[u]; 218 } 219 if (Deep[u] > Deep[v]) 220 swap(u, v); 221 if (Pre[u] + 1 <= Pre[v]) 222 Ans += Tree.Sum(1, Pre[u] + 1, Pre[v]); 223 return Ans; 224 } 225 226 int Query_Max(int u, int v) 227 { 228 int Ans(-INF), r1(Top[u]), r2(Top[v]); 229 while (r1 != r2) 230 { 231 if (Deep[r1] < Deep[r2]) 232 swap(u, v), swap(r1, r2); 233 Ans = max(Ans, Tree.Max(1, Pre[r1], Pre[u])); 234 u = Father[r1], r1 = Top[u]; 235 } 236 if (Deep[u] > Deep[v]) 237 swap(u, v); 238 if (Pre[u] + 1 <= Pre[v]) 239 Ans = max(Ans, Tree.Max(1, Pre[u] + 1, Pre[v])); 240 return Ans; 241 } 242 243 int Query_Min(int u, int v) 244 { 245 int Ans(INF), r1(Top[u]), r2(Top[v]); 246 while (r1 != r2) 247 { 248 if (Deep[r1] < Deep[r2]) 249 swap(u, v), swap(r1, r2); 250 Ans = min(Ans, Tree.Min(1, Pre[r1], Pre[u])); 251 u = Father[r1], r1 = Top[u]; 252 } 253 if (Deep[u] > Deep[v]) 254 swap(u, v); 255 if (Pre[u] + 1 <= Pre[v]) 256 Ans = min(Ans, Tree.Min(1, Pre[u] + 1, Pre[v])); 257 return Ans; 258 } 259 260 void dfs1(const int &u, const int &fa) 261 { 262 int v; 263 Size[u] = 1; 264 for (int i = Head[u];i;i = Next[i]) 265 { 266 v = To[i]; 267 if (v != fa) 268 { 269 Father[v] = u, Deep[v] = Deep[u] + 1, Cost[v] = Weight[i]; 270 dfs1(v, u); 271 Size[u] += Size[v]; 272 if (Size[v] > Size[Son[u]]) 273 Son[u] = v; 274 } 275 } 276 } 277 278 void dfs2(const int &u) 279 { 280 Pre[u] = ++Dfs_Clock, ID[Dfs_Clock] = u; 281 int v; 282 if (v = Son[u]) 283 { 284 Top[v] = Top[u]; 285 dfs2(v); 286 } 287 for (int i = Head[u];i;i = Next[i]) 288 { 289 v = To[i]; 290 if (v != Father[u] && v != Son[u]) 291 { 292 Top[v] = v; 293 dfs2(v); 294 } 295 } 296 } 297 298 void init() 299 { 300 int u, v, w; 301 Get_Val(N); 302 for (int i = 1;i != N;++i) 303 { 304 Get_Val(u), Get_Val(v), Get_Val(w), ++u, ++v; 305 Add_Edge(u, v, w), Add_Edge(v, u, w); 306 Edges[i].u = u, Edges[i].v = v; 307 } 308 Deep[root] = 1, dfs1(root, Father[root] = -1); 309 Top[root] = root, dfs2(root); 310 } 311 312 void work() 313 { 314 int M, u, v; 315 char op[10]; 316 Tree.build_tree(1, 1, N); 317 Get_Val(M); 318 while (M--) 319 { 320 scanf("%s", op), Get_Val(u), Get_Val(v); 321 if (*op == 'C') 322 { 323 if (Deep[Edges[u].u] < Deep[Edges[u].v]) 324 u = Edges[u].v; 325 else 326 u = Edges[u].u; 327 Tree.change(1, Pre[u], v); 328 } 329 else 330 if (*op == 'N') 331 ++u, ++v, Opposite(u, v); 332 else 333 if (*op == 'S') 334 ++u, ++v, printf("%d\n", Query_Sum(u, v)); 335 else 336 if (op[1] == 'A') 337 ++u, ++v, printf("%d\n", Query_Max(u, v)); 338 else 339 ++u, ++v, printf("%d\n", Query_Min(u, v)); 340 } 341 } 342 343 int main() 344 { 345 init(); 346 work(); 347 return 0; 348 }
BZOJ 2243 树剖+线段树区间合并,合并区间的时候还要判断左区间的右端点和右区间的左端点是否相同
#include <cstdio> #include <algorithm> #define LEFT (segt[cur].l) #define RIGHT (segt[cur].r) #define MID (segt[cur].mid) #define PIECE (segt[cur].piece) #define LC (segt[cur].lc) #define RC (segt[cur].rc) #define TAG (segt[cur].Tag) #define LCH (cur << 1) #define RCH ((cur << 1) | 1) using namespace std; void Get_Val(int &Ret) { Ret = 0; char ch; while (ch = getchar(), ch > '9' || ch < '0') ; do { (Ret *= 10) += ch - '0'; } while (ch = getchar(), ch >= '0' && ch <= '9'); } const int Max_N(100050); const int root(1); int N, M; int Colour[Max_N]; int Total, Head[Max_N], To[Max_N << 1], Next[Max_N << 1]; int Father[Max_N], Deep[Max_N], Size[Max_N], Son[Max_N], Top[Max_N]; int Tot, Pre[Max_N], ID[Max_N]; struct question { question(const int &a = 0, const int &b = 0, const int &c = 0) : piece(a), lc(b), rc(c) {} int piece, lc, rc; }; struct node { int l, r, mid; int piece; int lc, rc; int Tag; }; struct Segment_Tree { node segt[Max_N << 2]; void build_tree(const int&, const int&, const int&); inline void pushup(const int &cur) { PIECE = segt[LCH].piece + segt[RCH].piece - (segt[LCH].rc == segt[RCH].lc); LC = segt[LCH].lc, RC = segt[RCH].rc; } inline void pushdown(const int &cur) { if (TAG != -1) { segt[LCH].piece = segt[RCH].piece = 1; segt[LCH].lc = segt[LCH].rc = segt[RCH].lc = segt[RCH].rc = TAG; segt[LCH].Tag = segt[RCH].Tag = TAG; TAG = -1; } } void Set(const int&, const int&, const int&, const int&); question Query(const int&, const int&, const int&); }; Segment_Tree Tree; void Segment_Tree::build_tree(const int &cur, const int &l, const int &r) { LEFT = l, RIGHT = r, MID = l + ((r - l) >> 1), TAG = -1; if (l == r) { PIECE = 1; LC = RC = Colour[ID[l]]; return; } build_tree(LCH, l, MID), build_tree(RCH, MID + 1, r); pushup(cur); } void Segment_Tree::Set(const int &cur, const int &l, const int &r, const int &col) { if (l == LEFT && r == RIGHT) { PIECE = 1; LC = RC = TAG = col; return; } pushdown(cur); if (r <= MID) Set(LCH, l, r, col); else if (l > MID) Set(RCH, l, r, col); else { Set(LCH, l, MID, col); Set(RCH, MID + 1, r, col); } pushup(cur); } question Segment_Tree::Query(const int &cur, const int &l, const int &r) { if (l == LEFT && r == RIGHT) return question(PIECE, LC, RC); pushdown(cur); if (r <= MID) return Query(LCH, l, r); else if (l > MID) return Query(RCH, l, r); else { question ltree, rtree; ltree = Query(LCH, l, MID), rtree = Query(RCH, MID + 1, r); return question(ltree.piece + rtree.piece - (ltree.rc == rtree.lc), ltree.lc, rtree.rc); } } inline void Add_Edge(const int &s, const int &t) { ++Total; To[Total] = t; Next[Total] = Head[s], Head[s] = Total; } void dfs1(const int &u, const int &fa) { Size[u] = 1; int v; for (int i = Head[u];i;i = Next[i]) { v = To[i]; if (v != fa) { Father[v] = u, Deep[v] = Deep[u] + 1; dfs1(v, u); Size[u] += Size[v]; if (Size[v] > Size[Son[u]]) Son[u] = v; } } } void dfs2(const int &u) { int v; Pre[u] = ++Tot, ID[Tot] = u; if (v = Son[u]) { Top[v] = Top[u]; dfs2(v); } for (int i = Head[u];i;i = Next[i]) { v = To[i]; if (v != Father[u] && v != Son[u]) { Top[v] = v; dfs2(v); } } } void change(int u, int v, const int &c) { int r1(Top[u]), r2(Top[v]); while (r1 != r2) { if (Deep[r1] < Deep[r2]) swap(u, v), swap(r1, r2); Tree.Set(1, Pre[r1], Pre[u], c); u = Father[r1], r1 = Top[u]; } if (Deep[u] > Deep[v]) swap(u, v); Tree.Set(1, Pre[u], Pre[v], c); } int query(int u, int v) { int last_u_l(-1), last_u_r(-1), last_v_l(-1), last_v_r(-1); int now_u_l, now_u_r; int r1(Top[u]), r2(Top[v]), Ans(0); question now; while (r1 != r2) { if (Deep[r1] < Deep[r2]) swap(u, v), swap(r1, r2), swap(last_u_l, last_v_l), swap(last_u_r, last_v_r); now = Tree.Query(1, Pre[r1], Pre[u]); Ans += now.piece, now_u_l = now.lc, now_u_r = now.rc; Ans -= (now_u_r == last_u_l); last_u_l = now_u_l, last_u_r = now_u_r; u = Father[r1], r1 = Top[u]; } if (Deep[u] > Deep[v]) swap(u, v), swap(last_u_l, last_v_l), swap(last_u_r, last_v_r); now = Tree.Query(1, Pre[u], Pre[v]); Ans += now.piece, now_u_l = now.lc, now_u_r = now.rc; Ans -= (now_u_l == last_u_l) + (now_u_r == last_v_l); return Ans; } void init() { int x, y; Get_Val(N), Get_Val(M); for (int i = 1;i <= N;++i) Get_Val(Colour[i]); for (int i = 1;i != N;++i) { Get_Val(x), Get_Val(y); Add_Edge(x, y), Add_Edge(y, x); } Deep[root] = 1, dfs1(root, Father[root] = -1); Top[root] = root, dfs2(root); } void work() { char op[5]; int a, b, c; Tree.build_tree(1, 1, N); while (M--) { scanf("%s", op); if (*op == 'C') { Get_Val(a), Get_Val(b), Get_Val(c); change(a, b, c); } else { Get_Val(a), Get_Val(b); printf("%d\n", query(a, b)); } } } int main() { init(); work(); return 0; }
BZOJ 4034 不说了,裸题
1 #include <cstdio> 2 #include <algorithm> 3 #define LEFT (segt[cur].l) 4 #define RIGHT (segt[cur].r) 5 #define MID (segt[cur].mid) 6 #define SUM (segt[cur].Sum) 7 #define TAG (segt[cur].Tag) 8 #define LCH (cur << 1) 9 #define RCH ((cur << 1) | 1) 10 11 using namespace std; 12 13 template<typename Type> 14 void Get_Val(Type &Ret) 15 { 16 Ret = 0; 17 char ch; 18 bool Neg(false); 19 while (ch = getchar(), (ch > '9' || ch < '0') && ch != '-') 20 ; 21 if (ch == '-') 22 { 23 Neg = true; 24 while (ch = getchar(), ch > '9' || ch < '0') 25 ; 26 } 27 do 28 { 29 (Ret *= 10) += ch - '0'; 30 } 31 while (ch = getchar(), ch >= '0' && ch <= '9'); 32 Ret = (Neg ? -Ret : Ret); 33 } 34 35 const int Max_N(100050); 36 const int root(1); 37 38 int N, M; 39 long long int W[Max_N]; 40 int Total, Head[Max_N], To[Max_N << 1], Next[Max_N << 1]; 41 int Dfs_Clock, Father[Max_N], Deep[Max_N], Size[Max_N], Son[Max_N], Top[Max_N], Pre[Max_N], Post[Max_N], ID[Max_N]; 42 43 inline 44 void Add_Edge(const int &s, const int &t) 45 { 46 ++Total; 47 To[Total] = t; 48 Next[Total] = Head[s], Head[s] = Total; 49 } 50 51 struct node 52 { 53 int l, r, mid; 54 long long int Sum, Tag; 55 }; 56 57 struct Segment_Tree 58 { 59 node segt[Max_N << 2]; 60 void build_tree(const int&, const int&, const int&); 61 inline 62 void pushup(const int &cur) 63 { 64 SUM = segt[LCH].Sum + segt[RCH].Sum; 65 } 66 inline 67 void pushdown(const int &cur) 68 { 69 if (TAG) 70 { 71 segt[LCH].Sum += (segt[LCH].r - segt[LCH].l + 1) * TAG; 72 segt[RCH].Sum += (segt[RCH].r - segt[RCH].l + 1) * TAG; 73 segt[LCH].Tag += TAG, segt[RCH].Tag += TAG; 74 TAG = 0; 75 } 76 } 77 void Add(const int&, const int&, const int&, const long long int&); 78 long long int Sum(const int&, const int&, const int&); 79 }; 80 Segment_Tree Tree; 81 82 void Segment_Tree::build_tree(const int &cur, const int &l, const int &r) 83 { 84 LEFT = l, RIGHT = r, MID = l + ((r - l) >> 1), TAG = 0; 85 if (l == r) 86 { 87 SUM = W[ID[l]]; 88 return; 89 } 90 build_tree(LCH, l, MID), build_tree(RCH, MID + 1, r); 91 pushup(cur); 92 } 93 94 void Segment_Tree::Add(const int &cur, const int &l, const int &r, const long long int &x) 95 { 96 if (l == LEFT && r == RIGHT) 97 { 98 SUM += (RIGHT - LEFT + 1) * x; 99 TAG += x; 100 return; 101 } 102 pushdown(cur); 103 if (r <= MID) 104 Add(LCH, l, r, x); 105 else 106 if (l > MID) 107 Add(RCH, l, r, x); 108 else 109 Add(LCH, l, MID, x), Add(RCH, MID + 1, r, x); 110 pushup(cur); 111 } 112 113 long long int Segment_Tree::Sum(const int &cur, const int &l, const int &r) 114 { 115 if (l == LEFT && r == RIGHT) 116 return SUM; 117 pushdown(cur); 118 if (r <= MID) 119 return Sum(LCH, l, r); 120 else 121 if (l > MID) 122 return Sum(RCH, l, r); 123 else 124 return Sum(LCH, l, MID) + Sum(RCH, MID + 1, r); 125 } 126 127 void dfs1(const int &u, const int &fa) 128 { 129 Size[u] = 1; 130 int v; 131 for (int i = Head[u];i;i = Next[i]) 132 { 133 v = To[i]; 134 if (v != fa) 135 { 136 Father[v] = u, Deep[v] = Deep[u] + 1; 137 dfs1(v, u); 138 Size[u] += Size[v]; 139 if (Size[v] > Size[Son[u]]) 140 Son[u] = v; 141 } 142 } 143 } 144 145 void dfs2(const int &u) 146 { 147 Pre[u] = Post[u] = ++Dfs_Clock, ID[Dfs_Clock] = u; 148 int v; 149 if (v = Son[u]) 150 { 151 Top[v] = Top[u]; 152 dfs2(v); 153 Post[u] = max(Post[u], Post[v]); 154 } 155 for (int i = Head[u];i;i = Next[i]) 156 { 157 v = To[i]; 158 if (v != Father[u] && v != Son[u]) 159 { 160 Top[v] = v; 161 dfs2(v); 162 Post[u] = max(Post[u], Post[v]); 163 } 164 } 165 } 166 167 void init() 168 { 169 Get_Val(N), Get_Val(M); 170 int fr, to; 171 for (int i = 1;i <= N;++i) 172 Get_Val(W[i]); 173 for (int i = 1;i != N;++i) 174 { 175 Get_Val(fr), Get_Val(to); 176 Add_Edge(fr, to), Add_Edge(to, fr); 177 } 178 Deep[root] = 1, dfs1(root, Father[root] = -1); 179 Top[root] = root, dfs2(root); 180 } 181 182 long long int Query(int u, int v) 183 { 184 long long int Sum(0), r1(Top[u]), r2(Top[v]); 185 while (r1 != r2) 186 { 187 if (Deep[r1] < Deep[r2]) 188 swap(u, v), swap(r1, r2); 189 Sum += Tree.Sum(1, Pre[r1], Pre[u]); 190 u = Father[r1], r1 = Top[u]; 191 } 192 if (Deep[u] > Deep[v]) 193 swap(u, v); 194 return Sum + Tree.Sum(1, Pre[u], Pre[v]); 195 } 196 197 void work() 198 { 199 Tree.build_tree(1, 1, N); 200 int op, x, a; 201 while (M--) 202 { 203 Get_Val(op); 204 if (op == 1) 205 { 206 Get_Val(x), Get_Val(a); 207 Tree.Add(1, Pre[x], Pre[x], a); 208 } 209 if (op == 2) 210 { 211 Get_Val(x), Get_Val(a); 212 Tree.Add(1, Pre[x], Post[x], a); 213 } 214 if (op == 3) 215 { 216 Get_Val(x); 217 printf("%lld\n", Query(x, root)); 218 } 219 } 220 } 221 222 int main() 223 { 224 init(); 225 work(); 226 return 0; 227 }
BZOJ 4196 裸题。。觉得noi考这种数据结构裸题很没有意思啊
1 #include <queue> 2 #include <cstdio> 3 #include <cstring> 4 #include <algorithm> 5 #define LEFT (segt[cur].l) 6 #define RIGHT (segt[cur].r) 7 #define MID (segt[cur].mid) 8 #define ZERO (segt[cur].Sum[0]) 9 #define ONE (segt[cur].Sum[1]) 10 #define TAG (segt[cur].Tag) 11 #define LCH (cur << 1) 12 #define RCH ((cur << 1) | 1) 13 14 using namespace std; 15 16 const int Max_N(100050); 17 const int root(1); 18 19 void Get_Val(int &Ret) 20 { 21 Ret = 0; 22 char ch; 23 while (ch = getchar(), ch > '9' || ch < '0') 24 ; 25 do 26 { 27 (Ret *= 10) += ch - '0'; 28 } 29 while (ch = getchar(), ch >= '0' && ch <= '9'); 30 } 31 32 int N; 33 int Total, Head[Max_N], To[Max_N], Next[Max_N]; 34 int Father[Max_N], Deep[Max_N], Size[Max_N], Son[Max_N], Top[Max_N]; 35 int Dfs_Clock, Pre[Max_N], Post[Max_N]; 36 int Degree[Max_N]; 37 38 struct node 39 { 40 int l, r, mid; 41 int Sum[2], Tag; 42 }; 43 44 struct Segment_Tree 45 { 46 node segt[Max_N << 2]; 47 void build_tree(const int&, const int&, const int&); 48 inline 49 void pushup(const int &cur) 50 { 51 ZERO = segt[LCH].Sum[0] + segt[RCH].Sum[0]; 52 ONE = segt[LCH].Sum[1] + segt[RCH].Sum[1]; 53 } 54 inline 55 void pushdown(const int &cur) 56 { 57 if (TAG != -1) 58 { 59 if (TAG) 60 { 61 segt[LCH].Sum[1] = segt[LCH].r - segt[LCH].l + 1; 62 segt[RCH].Sum[1] = segt[RCH].r - segt[RCH].l + 1; 63 segt[LCH].Sum[0] = segt[RCH].Sum[0] = 0; 64 } 65 else 66 { 67 segt[LCH].Sum[0] = segt[LCH].r - segt[LCH].l + 1; 68 segt[RCH].Sum[0] = segt[RCH].r - segt[RCH].l + 1; 69 segt[LCH].Sum[1] = segt[RCH].Sum[1] = 0; 70 } 71 segt[LCH].Tag = segt[RCH].Tag = TAG; 72 TAG = -1; 73 } 74 } 75 int Set(const int&, const int&, const int&, const int&); 76 }; 77 Segment_Tree Tree; 78 79 void Segment_Tree::build_tree(const int &cur, const int &l, const int &r) 80 { 81 LEFT = l, RIGHT = r, MID = l + ((r - l) >> 1); 82 ONE = 0, ZERO = r - l + 1, TAG = -1; 83 if (l == r) 84 return; 85 build_tree(LCH, l, MID), build_tree(RCH, MID + 1, r); 86 } 87 88 int Segment_Tree::Set(const int &cur, const int &l, const int &r, const int &x) 89 { 90 int Ans; 91 if (l == LEFT && r == RIGHT) 92 { 93 Ans = segt[cur].Sum[1 - x]; 94 if (x) 95 ONE = RIGHT - LEFT + 1, ZERO = 0; 96 else 97 ZERO = RIGHT - LEFT + 1, ONE = 0; 98 TAG = x; 99 return Ans; 100 } 101 pushdown(cur); 102 if (r <= MID) 103 { 104 Ans = Set(LCH, l, r, x); 105 pushup(cur); 106 } 107 else 108 if (l > MID) 109 { 110 Ans = Set(RCH, l, r, x); 111 pushup(cur); 112 } 113 else 114 { 115 Ans = Set(LCH, l, MID, x) + Set(RCH, MID + 1, r, x); 116 pushup(cur); 117 } 118 return Ans; 119 } 120 121 inline 122 void Add_Edge(const int &s, const int &t) 123 { 124 ++Total; 125 To[Total] = t; 126 Next[Total] = Head[s], Head[s] = Total; 127 } 128 129 void bfs1() 130 { 131 queue<int> Q; 132 int u, fa; 133 for (int i = 1;i <= N;++i) 134 if (!Degree[i]) 135 Q.push(i); 136 while (Q.size()) 137 { 138 u = Q.front(), Q.pop(); 139 fa = Father[u]; 140 ++Size[u]; 141 if (u != root) 142 { 143 Size[fa] += Size[u]; 144 if (Size[u] > Size[Son[fa]]) 145 Son[fa] = u; 146 Add_Edge(fa, u); 147 if ((--Degree[fa]) == 0) 148 Q.push(fa); 149 } 150 } 151 } 152 153 void dfs2(const int &u) 154 { 155 int v; 156 Pre[u] = Post[u] = ++Dfs_Clock; 157 if (v = Son[u]) 158 { 159 Top[v] = Top[u]; 160 Deep[v] = Deep[u] + 1; 161 dfs2(v); 162 Post[u] = Post[v]; 163 } 164 for (int i = Head[u];i;i = Next[i]) 165 if ((v = To[i]) != Son[u]) 166 { 167 Top[v] = v; 168 Deep[v] = Deep[u] + 1; 169 dfs2(v); 170 Post[u] = Post[v]; 171 } 172 } 173 174 void init() 175 { 176 Get_Val(N); 177 for (int i = 2;i <= N;++i) 178 Get_Val(Father[i]), ++Father[i], ++Degree[Father[i]]; 179 Father[root] = -1, bfs1(); 180 Deep[root] = 1, Top[root] = root, dfs2(root); 181 } 182 183 int Query(int u, int v) 184 { 185 int Ans(0), r1(Top[u]), r2(Top[v]); 186 while (r1 != r2) 187 { 188 if (Deep[r1] < Deep[r2]) 189 swap(r1, r2), swap(u, v); 190 Ans += Tree.Set(1, Pre[r1], Pre[u], 1); 191 u = Father[r1], r1 = Top[u]; 192 } 193 if (Deep[u] > Deep[v]) 194 swap(u, v); 195 return Ans + Tree.Set(1, Pre[u], Pre[v], 1); 196 } 197 198 void work() 199 { 200 int Q, x; 201 Get_Val(Q); 202 char op[20]; 203 Tree.build_tree(1, 1, N); 204 while (Q--) 205 { 206 scanf("%s", op); 207 Get_Val(x); 208 ++x; 209 if (*op == 'i') 210 printf("%d\n", Query(x, root)); 211 else 212 printf("%d\n", Tree.Set(1, Pre[x], Post[x], 0)); 213 } 214 } 215 216 int main() 217 { 218 init(); 219 work(); 220 return 0; 221 }