結果
問題 | No.529 帰省ラッシュ |
ユーザー | penguinshunya |
提出日時 | 2019-07-10 14:05:30 |
言語 | C++17 (gcc 13.3.0 + boost 1.87.0) |
結果 |
AC
|
実行時間 | 733 ms / 4,500 ms |
コード長 | 5,688 bytes |
コンパイル時間 | 3,203 ms |
コンパイル使用メモリ | 237,900 KB |
最終ジャッジ日時 | 2025-01-07 06:33:06 |
ジャッジサーバーID (参考情報) |
judge3 / judge5 |
(要ログイン)
テストケース
テストケース表示入力 | 結果 | 実行時間 実行使用メモリ |
---|---|---|
testcase_00 | AC | 1 ms
6,820 KB |
testcase_01 | AC | 1 ms
6,816 KB |
testcase_02 | AC | 2 ms
6,820 KB |
testcase_03 | AC | 2 ms
6,824 KB |
testcase_04 | AC | 9 ms
6,824 KB |
testcase_05 | AC | 10 ms
6,820 KB |
testcase_06 | AC | 10 ms
6,820 KB |
testcase_07 | AC | 10 ms
6,820 KB |
testcase_08 | AC | 605 ms
24,192 KB |
testcase_09 | AC | 622 ms
24,688 KB |
testcase_10 | AC | 703 ms
31,408 KB |
testcase_11 | AC | 673 ms
31,436 KB |
testcase_12 | AC | 564 ms
26,196 KB |
testcase_13 | AC | 539 ms
53,424 KB |
testcase_14 | AC | 599 ms
28,140 KB |
testcase_15 | AC | 733 ms
34,984 KB |
testcase_16 | AC | 729 ms
35,180 KB |
testcase_17 | AC | 688 ms
45,808 KB |
testcase_18 | AC | 709 ms
46,340 KB |
testcase_19 | AC | 709 ms
43,144 KB |
ソースコード
#include <bits/stdc++.h> using namespace std; struct UnWeightedGraph { struct Edge { int to; Edge(int to) : to(to) {} }; vector<vector<Edge>> edges; int n; UnWeightedGraph(int n) : n(n) { edges.assign(n, vector<Edge>()); } void add_edge(int v, int u) { edges[v].emplace_back(u); } vector<Edge> &operator[](int x) { return edges[x]; } }; template <typename Graph> struct LowLink { using A = vector<int>; using B = vector<pair<int, int>>; Graph &g; int n; vector<bool> seen; vector<int> ord, low; A articulation; B bridge; LowLink(Graph &g) : g(g), n(g.n) {} void dfs(int v, int p, int &k) { seen[v] = true; ord[v] = k++; low[v] = ord[v]; bool is = false; int cnt = 0; for (auto e : g[v]) { int u = e.to; if (seen[u]) { if (u != p) low[v] = min(low[v], ord[u]); continue; } cnt++; dfs(u, v, k); low[v] = min(low[v], low[u]); if (p != -1 && ord[v] <= low[u]) is = true; if (ord[v] < low[u]) bridge.emplace_back(minmax(v, u)); } if (p == -1 && cnt > 1) is = true; if (is) articulation.push_back(v); } pair<A, B> build() { seen.assign(n, false); ord.assign(n, 0); low.assign(n, 0); articulation.clear(); bridge.clear(); int k = 0; dfs(0, -1, k); return pair<A, B>(articulation, bridge); } pair<A, B> operator()() { return build(); } }; template <typename Graph> struct TwoEdgeConnectedComponent { using B = vector<pair<int, int>>; Graph &g; int n; vector<bool> seen; vector<int> group; set<pair<int, int>> b; TwoEdgeConnectedComponent(Graph &g) : g(g), n(g.n), group(n) {} void dfs(int v, int p, int k) { seen[v] = true; group[v] = k; for (auto e : g[v]) { int u = e.to; if (u == p) continue; if (seen[u]) continue; if (b.find(minmax(v, u)) != b.end()) continue; dfs(u, v, k); } } vector<int> decompose(B bridge) { seen.assign(n, false); b.clear(); b.insert(bridge.begin(), bridge.end()); int k = 0; for (int i = 0; i < n; i++) { if (seen[i]) continue; dfs(i, -1, k++); } return group; } vector<int> operator()(B bridge) { return decompose(bridge); } }; template <typename T> struct SegmentTree { using F = function<T(T, T)>; vector<T> v; F f; T e; int n; SegmentTree(int size, F f, T e) : f(f), e(e) { n = 1; while (n < size) n <<= 1; v.resize(n * 2, e); } void set(int k, T x) { v[k + n] = x; } void build() { for (int i = n - 1; i > 0; i--) { v[i] = f(v[i * 2 + 0], v[i * 2 + 1]); } } void update(int k, T x) { v[k += n] = x; while (k >>= 1) v[k] = f(v[k * 2 + 0], v[k * 2 + 1]); } T query(int a, int b) { T l = e, r = e; for (a += n, b += n; a < b; a >>= 1, b >>= 1) { if (a & 1) l = f(l, v[a++]); if (b & 1) r = f(v[--b], r); } return f(l, r); } }; template <typename Graph> struct HeavyLightDecomposition { Graph &g; int n; vector<int> par; vector<int> next; vector<int> vid; vector<int> head; HeavyLightDecomposition(Graph &g) : g(g), n(g.n), par(n), next(n, -1), vid(n), head(n) {} int dfs(int v = 0, int p = -1) { par[v] = p; int mx = 0, ret = 1; for (auto e : g[v]) { if (e.to == p) continue; int r = dfs(e.to, v); ret += r; if (mx < r) mx = r, next[v] = e.to; } return ret; } void bfs(int r = 0) { int k = 0; queue<int> q; q.emplace(r); while (!q.empty()) { auto h = q.front(); q.pop(); for (int v = h; v != -1; v = next[v]) { vid[v] = k++; head[v] = h; for (auto e : g[v]) { if (e.to == par[v] || e.to == next[v]) continue; q.emplace(e.to); } } } } vector<int> build() { dfs(); bfs(); return vid; } void for_each(int v, int u, const function<void(int, int)> &f) { while (true) { if (vid[u] > vid[v]) swap(v, u); f(max(vid[head[v]], vid[u]), vid[v]); if (head[u] != head[v]) v = par[head[v]]; else break; } } }; int main() { int N, M, Q; cin >> N >> M >> Q; auto graph = UnWeightedGraph(N); for (int i = 0; i < M; i++) { int v, u; cin >> v >> u; v--, u--; graph.add_edge(v, u); graph.add_edge(u, v); } auto low = LowLink<UnWeightedGraph>(graph); auto two = TwoEdgeConnectedComponent<UnWeightedGraph>(graph); auto lll = low.build(); auto dict = two.decompose(lll.second); int V = *max_element(dict.begin(), dict.end()) + 1; auto hraph = UnWeightedGraph(V); for (int i = 0; i < N; i++) { for (auto e : graph[i]) { if (dict[i] == dict[e.to]) continue; hraph.add_edge(dict[i], dict[e.to]); } } auto hld = HeavyLightDecomposition<UnWeightedGraph>(hraph); auto vid = hld.build(); auto seg = SegmentTree<int>(V, [](int a, int b) { return max(a, b); }, -1); vector<priority_queue<int>> q(V); unordered_map<int, int> ma; for (int i = 0; i < Q; i++) { int n; cin >> n; if (n == 1) { int u, w; cin >> u >> w; u--; u = vid[dict[u]]; if (q[u].empty() || q[u].top() < w) { seg.update(u, w); } q[u].emplace(w); ma[w] = u; } else { int s, t; cin >> s >> t; s--, t--; int ans = -1; s = dict[s], t = dict[t]; hld.for_each(s, t, [&](int l, int r) { ans = max(ans, seg.query(l, r + 1)); }); cout << ans << '\n'; if (ans != -1) { int u = ma[ans]; q[u].pop(); seg.update(u, q[u].size() > 0 ? q[u].top() : -1); } } } return 0; }