結果
問題 | No.399 動的な領主 |
ユーザー | Suikaba |
提出日時 | 2018-09-26 00:09:24 |
言語 | C++17 (gcc 12.3.0 + boost 1.83.0) |
結果 |
AC
|
実行時間 | 223 ms / 2,000 ms |
コード長 | 4,723 bytes |
コンパイル時間 | 2,567 ms |
コンパイル使用メモリ | 217,288 KB |
実行使用メモリ | 12,396 KB |
最終ジャッジ日時 | 2024-10-07 05:42:38 |
合計ジャッジ時間 | 6,063 ms |
ジャッジサーバーID (参考情報) |
judge2 / judge4 |
(要ログイン)
テストケース
テストケース表示入力 | 結果 | 実行時間 実行使用メモリ |
---|---|---|
testcase_00 | AC | 2 ms
6,816 KB |
testcase_01 | AC | 2 ms
6,816 KB |
testcase_02 | AC | 1 ms
6,820 KB |
testcase_03 | AC | 2 ms
6,816 KB |
testcase_04 | AC | 3 ms
6,816 KB |
testcase_05 | AC | 17 ms
6,816 KB |
testcase_06 | AC | 212 ms
11,880 KB |
testcase_07 | AC | 219 ms
11,996 KB |
testcase_08 | AC | 218 ms
11,964 KB |
testcase_09 | AC | 217 ms
11,932 KB |
testcase_10 | AC | 3 ms
6,816 KB |
testcase_11 | AC | 15 ms
6,816 KB |
testcase_12 | AC | 182 ms
12,396 KB |
testcase_13 | AC | 182 ms
12,380 KB |
testcase_14 | AC | 126 ms
11,848 KB |
testcase_15 | AC | 129 ms
11,848 KB |
testcase_16 | AC | 149 ms
11,844 KB |
testcase_17 | AC | 223 ms
11,932 KB |
testcase_18 | AC | 222 ms
12,064 KB |
ソースコード
#include <bits/stdc++.h> using namespace std; using ll = long long; class heavy_light_decomposition { public: heavy_light_decomposition(int n_) : n(n_), g(n), par(n), head(n), in(n), out(n) {} heavy_light_decomposition(std::vector<std::vector<int>> const& g_) : n(g_.size()), g(g_), par(n), head(n), in(n), out(n) {} void add_edge(int u, int v) { g[u].push_back(v); g[v].push_back(u); } void build(int rt = 0) { dfs1(rt); head[rt] = rt; dfs2(rt); } int get_id(int v) const { return in[v]; } void path_query(int u, int v, std::function<void(int, int)> f) { // vertex while(true) { if(in[u] > in[v]) std::swap(u, v); f(std::max(in[head[v]], in[u]), in[v] + 1); if(head[u] == head[v]) return; v = par[head[v]]; } } void edge_path_query(int u, int v, std::function<void(int, int)> f) { while(true) { if(in[u] > in[v]) std::swap(u, v); if(head[u] != head[v]) { f(std::max(in[head[v]], in[u]), in[v] + 1); v = par[head[v]]; } else { if(u != v) f(in[u] + 1, in[v] + 1); break; } } } void subtree_query(int v, std::function<void(int, int)> f) { f(in[v], out[v]); } int get_lca(int u, int v) const { while(true) { if(in[u] > in[v]) std::swap(u, v); if(head[u] == head[v]) return u; v = par[head[v]]; } } private: void dfs1(int rt) { std::vector<int> sz(n, 1), iter(n); std::vector<std::pair<int, int>> st = {{rt, -1}}; st.reserve(n); while(!st.empty()) { const int v = st.back().first; if(iter[v] < (int)g[v].size()) { if(g[v][iter[v]] != st.back().second) { st.emplace_back(g[v][iter[v]], v); } ++iter[v]; continue; } par[v] = st.back().second; if(par[v] != -1) { const int pidx = std::find(std::begin(g[v]), std::end(g[v]), par[v]) - std::begin(g[v]); std::swap(g[v].back(), g[v][pidx]); g[v].pop_back(); } for(auto& u : g[v]) { sz[v] += sz[u]; if(sz[u] > sz[g[v].front()]) std::swap(u, g[v].front()); } st.pop_back(); } } void dfs2(int rt) { int it = 0; std::vector<int> st = {rt}, iter(n); st.reserve(n); while(!st.empty()) { const int v = st.back(); if(!iter[v]) in[v] = it++; // first visit if(iter[v] < (int)g[v].size()) { int u = g[v][iter[v]]; head[u] = iter[v] ? u : head[v]; st.push_back(u); ++iter[v]; continue; } out[v] = it; st.pop_back(); } } private: const int n; std::vector<std::vector<int>> g; std::vector<int> par, head, in, out; }; template <typename T> class fenwick_tree { public: fenwick_tree(int n_) : n(n_), dat(n, 0) {} void add(int i, T value) { for(; i < n; i |= i + 1) { dat[i] += value; } } T sum(int i) const { T res = 0; for(; i >= 0; i = (i & (i + 1)) - 1) { res += dat[i]; } return res; } // [l, r) T sum(int l, int r) const { return sum(r - 1) - sum(l - 1); } private: const int n; std::vector<T> dat; }; template <typename T> class range_add_range_sum { public: range_add_range_sum(int n_) : n(n_), bit0(n), bit1(n) {} void add(int l, int r, T val) { bit0.add(l, -val * l); bit1.add(l, val); bit0.add(r, val * r); bit1.add(r, -val); } T sum(int p) const { return bit1.sum(p) * (p + 1) + bit0.sum(p); } T sum(int l, int r) const { return sum(r - 1) - sum(l - 1); } private: const int n; fenwick_tree<T> bit0, bit1; }; int main() { int n; cin >> n; heavy_light_decomposition hld(n); for(int i = 0; i < n - 1; ++i) { int u, v; cin >> u >> v; hld.add_edge(u - 1, v - 1); } hld.build(); int q; cin >> q; range_add_range_sum<ll> rars(n); while(q--) { int a, b; cin >> a >> b; hld.path_query(a - 1, b - 1, [&] (int l, int r) { rars.add(l, r, 1); }); } ll ans = 0; for(int i = 0; i < n; ++i) { const ll t = rars.sum(i, i + 1); ans += t * (t + 1) / 2; } cout << ans << endl; }