結果
問題 | No.899 γatheree |
ユーザー | niuez |
提出日時 | 2020-03-24 16:22:49 |
言語 | C++14 (gcc 13.3.0 + boost 1.87.0) |
結果 |
AC
|
実行時間 | 653 ms / 2,000 ms |
コード長 | 6,061 bytes |
コンパイル時間 | 2,681 ms |
コンパイル使用メモリ | 187,792 KB |
実行使用メモリ | 16,692 KB |
最終ジャッジ日時 | 2024-12-31 15:11:20 |
合計ジャッジ時間 | 14,264 ms |
ジャッジサーバーID (参考情報) |
judge4 / judge2 |
(要ログイン)
テストケース
テストケース表示入力 | 結果 | 実行時間 実行使用メモリ |
---|---|---|
testcase_00 | AC | 2 ms
6,816 KB |
testcase_01 | AC | 2 ms
6,816 KB |
testcase_02 | AC | 3 ms
6,816 KB |
testcase_03 | AC | 3 ms
6,816 KB |
testcase_04 | AC | 2 ms
6,816 KB |
testcase_05 | AC | 2 ms
6,816 KB |
testcase_06 | AC | 641 ms
16,476 KB |
testcase_07 | AC | 653 ms
16,352 KB |
testcase_08 | AC | 548 ms
16,476 KB |
testcase_09 | AC | 554 ms
16,472 KB |
testcase_10 | AC | 562 ms
16,476 KB |
testcase_11 | AC | 568 ms
16,476 KB |
testcase_12 | AC | 572 ms
16,604 KB |
testcase_13 | AC | 601 ms
16,476 KB |
testcase_14 | AC | 596 ms
16,348 KB |
testcase_15 | AC | 573 ms
16,472 KB |
testcase_16 | AC | 576 ms
16,352 KB |
testcase_17 | AC | 577 ms
16,480 KB |
testcase_18 | AC | 563 ms
16,476 KB |
testcase_19 | AC | 570 ms
16,476 KB |
testcase_20 | AC | 590 ms
16,604 KB |
testcase_21 | AC | 481 ms
16,564 KB |
testcase_22 | AC | 473 ms
16,568 KB |
testcase_23 | AC | 469 ms
16,692 KB |
ソースコード
#include <vector> #include <iostream> struct bfs_euler_tour { int N; std::vector<std::vector<int>> G; std::vector<int> in; std::vector<int> out; std::vector<int> para; std::vector<int> inv_para; std::vector<int> dep; std::vector<int> par; std::vector<int> start; int cnt; int D; bfs_euler_tour(int N): N(N), G(N), in(N), out(N), para(N, -1), inv_para(N, -1), dep(N), par(N) {} void add_edge(int a, int b) { G[a].push_back(b); G[b].push_back(a); } void dfs(int v, int f, int depth) { D = std::max(D, depth); par[v] = f; dep[v] = depth; in[v] = cnt++; for(auto t: G[v]) { if(t == f) continue; dfs(t, v, depth + 1); } out[v] = cnt; } void build(int r) { cnt = 0; D = 0; dfs(r, -1, 0); D++; cnt = 0; std::vector<int> que(N); int ql = 0; int qr = 0; que[qr++] = r; start.resize(D + 1); for(int d = 0; ql < qr; d++) { int r = qr; start[d] = cnt; while(ql < r) { int v = que[ql++]; inv_para[v] = cnt; para[cnt++] = v; for(auto t: G[v]) { if(in[v] < in[t]) { que[qr++] = t; } } } } start[D] = cnt; } int para_lower_bound(int l, int r, int i) { while(r - l > 1) { int m = (l + r) >> 1; if(i <= in[para[m]]) { r = m; } else { l = m; } } return r; } std::pair<int, int> range(int v, int d) { if(dep[v] + d < D) { int l = start[dep[v] + d]; int r = start[dep[v] + d + 1]; return { para_lower_bound(l - 1, r, in[v]), para_lower_bound(l - 1, r, out[v]) }; } else { return { 0, 0 }; } } }; #include <bits/stdc++.h> using namespace std; using i64 = long long; #define rep(i,s,e) for(i64 (i) = (s);(i) < (e);(i)++) #define all(x) x.begin(),x.end() template<class T> static inline std::vector<T> ndvec(size_t&& n, T val) noexcept { return std::vector<T>(n, std::forward<T>(val)); } template<class... Tail> static inline auto ndvec(size_t&& n, Tail&&... tail) noexcept { return std::vector<decltype(ndvec(std::forward<Tail>(tail)...))>(n, ndvec(std::forward<Tail>(tail)...)); } template<class T, class Cond> struct chain { Cond cond; chain(Cond cond) : cond(cond) {} bool operator()(T& a, const T& b) const { if(cond(a, b)) { a = b; return true; } return false; } }; template<class T, class Cond> chain<T, Cond> make_chain(Cond cond) { return chain<T, Cond>(cond); } #include <vector> #include <set> #include <iostream> using i64 = long long; struct lazy_segment_tree { using T = i64; using L = i64; static inline T t_ide() { return 0; } static inline L l_ide() { return 0; } static inline T ope(const T& a, const T& b) { return a + b; } static inline L lazy_ope(const L& a, const L& b) { return b; } static inline T effect(const T& t, const L& l) { return l; } int n, h; std::vector<T> node; std::vector<L> lazy; std::vector<bool> flag; lazy_segment_tree(int N) { n = 1; h = 1; while(n < N) n <<= 1, h++; node.resize(n << 1, t_ide()); lazy.resize(n << 1, l_ide()); flag.resize(n << 1, false); } lazy_segment_tree(const std::vector<T>& init) { n = 1; h = 1; while(n < init.size()) n <<= 1, h++; node.resize(n << 1, t_ide()); lazy.resize(n << 1, l_ide()); flag.resize(n << 1, false); for(int i = 0;i < init.size();i++) node[i + n] = init[i]; for(int i = n; i --> 1;) node[i] = ope(node[(i << 1)], node[(i << 1) + 1]); } inline void eff(int k, L x) { if(k < n << 1) { lazy[k] = lazy_ope(lazy[k], x); flag[k] = true; } } inline T eval(int k) const { return flag[k] ? effect(node[k], lazy[k]) : node[k]; } inline void push(int k) { if(flag[k]) { node[k] = eval(k); eff(k << 1, lazy[k]); eff((k << 1) | 1, lazy[k]); lazy[k] = l_ide(); flag[k] = false; } } inline void infuse(int k) { k = k >> __builtin_ctz(k); while((k >>= 1)) node[k] = ope(eval(k << 1), eval((k << 1) + 1)); } inline void infiltrate(int k) { if(k == n << 1) return; int kc = __builtin_ctz(k); for(int i = h; i --> kc;) push(k >> i); } inline void infiltrate(int l, int r) { if(l == r) return; if(r == n << 1) infiltrate(l); else { int hh = h; int x = l ^ r; for(; !(x >> --hh);) push(l >> hh); int lc = __builtin_ctz(l); for(int i = hh + 1; i --> lc;) push(l >> i); int rc = __builtin_ctz(r); for(int i = hh + 1; i --> rc;) push(r >> i); } } void update(int a, int b, L x) { int l = a + n; int r = b + n; infiltrate(l, r); while(l < r) { if(l & 1) eff(l++, x); if(r & 1) eff(--r, x); l >>= 1; r >>= 1; } infuse(a + n); infuse(b + n); } T sum(int l, int r) { l += n; r += n; infiltrate(l, r); T lx = t_ide(); T rx = t_ide(); while(l < r) { if(l & 1) lx = ope(lx, eval(l++)); if(r & 1) rx = ope(eval(--r), rx); l >>= 1; r >>= 1; } return ope(lx, rx); } }; int main() { cin.tie(nullptr); std::ios::sync_with_stdio(false); int N; cin >> N; bfs_euler_tour eul(N); rep(i,0,N - 1) { int a, b; cin >> a >> b; eul.add_edge(a, b); } eul.build(0); vector<int> A(N); vector<i64> init(N); rep(i,0,N) { cin >> A[i]; } rep(i,0,N) { init[i] = A[eul.para[i]]; } lazy_segment_tree seg(init); int Q; cin >> Q; i64 sum = 0; auto func = [&](int v, int d) { int l, r; std::tie(l, r) = eul.range(v, d); sum += seg.sum(l, r); seg.update(l, r, 0); }; while(Q--){ int v; cin >> v; sum = 0; if(eul.par[v] != -1) { int p = eul.par[v]; if(eul.par[p] != -1) { func(eul.par[p], 0); } func(p, 0); func(p, 1); } else { func(v, 0); } func(v, 1); func(v, 2); cout << sum << "\n"; seg.update(eul.inv_para[v], eul.inv_para[v] + 1, sum); } }