結果
問題 | No.1976 Cut then Connect |
ユーザー | shino16 |
提出日時 | 2022-05-31 04:49:16 |
言語 | C++17 (gcc 13.3.0 + boost 1.87.0) |
結果 |
AC
|
実行時間 | 110 ms / 2,000 ms |
コード長 | 3,771 bytes |
コンパイル時間 | 3,811 ms |
コンパイル使用メモリ | 236,328 KB |
実行使用メモリ | 24,704 KB |
最終ジャッジ日時 | 2024-11-24 12:13:43 |
合計ジャッジ時間 | 5,828 ms |
ジャッジサーバーID (参考情報) |
judge2 / judge5 |
(要ログイン)
テストケース
テストケース表示入力 | 結果 | 実行時間 実行使用メモリ |
---|---|---|
testcase_00 | AC | 5 ms
10,648 KB |
testcase_01 | AC | 5 ms
10,744 KB |
testcase_02 | AC | 102 ms
24,072 KB |
testcase_03 | AC | 56 ms
18,660 KB |
testcase_04 | AC | 22 ms
13,568 KB |
testcase_05 | AC | 47 ms
17,604 KB |
testcase_06 | AC | 72 ms
21,108 KB |
testcase_07 | AC | 48 ms
17,740 KB |
testcase_08 | AC | 108 ms
24,704 KB |
testcase_09 | AC | 83 ms
22,628 KB |
testcase_10 | AC | 25 ms
14,172 KB |
testcase_11 | AC | 102 ms
24,592 KB |
testcase_12 | AC | 7 ms
11,196 KB |
testcase_13 | AC | 77 ms
21,120 KB |
testcase_14 | AC | 70 ms
20,376 KB |
testcase_15 | AC | 60 ms
20,112 KB |
testcase_16 | AC | 86 ms
23,304 KB |
testcase_17 | AC | 32 ms
15,488 KB |
testcase_18 | AC | 10 ms
11,684 KB |
testcase_19 | AC | 73 ms
21,168 KB |
testcase_20 | AC | 110 ms
24,704 KB |
testcase_21 | AC | 57 ms
19,056 KB |
testcase_22 | AC | 4 ms
10,624 KB |
testcase_23 | AC | 4 ms
10,624 KB |
testcase_24 | AC | 4 ms
10,620 KB |
testcase_25 | AC | 5 ms
10,720 KB |
testcase_26 | AC | 4 ms
10,624 KB |
testcase_27 | AC | 4 ms
10,752 KB |
testcase_28 | AC | 5 ms
10,576 KB |
testcase_29 | AC | 4 ms
10,624 KB |
testcase_30 | AC | 4 ms
10,624 KB |
testcase_31 | AC | 4 ms
10,624 KB |
testcase_32 | AC | 5 ms
10,592 KB |
ソースコード
// 制約チェック #line 2 "lib/prelude.hpp" #ifndef LOCAL #pragma GCC optimize("O3,unroll-loops") #pragma GCC target("avx2") #endif #include <bits/stdc++.h> using namespace std; using ll = long long; #define rep2(i, m, n) for (auto i = (m); i < (n); i++) #define rep(i, n) rep2(i, 0, n) #define repr2(i, m, n) for (auto i = (n); i-- > (m);) #define repr(i, n) repr2(i, 0, n) #define all(x) begin(x), end(x) #line 3 "lib/ds/dsu.hpp" class dsu { public: dsu(int n) : par(n, -1), count_(n) {} int size() const { return par.size(); } int count() const { return count_; } void clear() { fill(par.begin(), par.end(), -1); count_ = size(); } int find(int x) { return par[x] < 0 ? x : par[x] = (int)find(par[x]); } bool same(int x, int y) { return find(x) == find(y); } bool unite(int x, int y) { x = find(x), y = find(y); if (x == y) return false; if (par[x] > par[y]) swap(x, y); par[x] += par[y], par[y] = (int)x; count_--; return true; } vector<vector<int>> groups() { vector<vector<int>> res(size()); for (int x = 0; x < size(); x++) res[find(x)].push_back(x); res.erase(remove_if(all(res), [](const auto& v) { return v.empty(); }), res.end()); return res; } private: vector<int> par; int count_; }; #line 3 "main.cpp" const int MaxN = 100000; int n; vector<int> G[MaxN]; // paths[v] = {(v-u-で始まる最長パス長, u)} 降順 vector<pair<int, int>> paths[MaxN]; // longest[v] = {(uの部分木内の最長パス長, u)} 降順 vector<pair<int, int>> longest[MaxN]; int ans; void dfs1(int v, int p) { for (auto u : G[v]) if (u != p) { dfs1(u, v); paths[v].emplace_back(paths[u][0].first + 1, u); longest[v].emplace_back( max(longest[u][0].first, paths[u][0].first + paths[u][1].first), u); } paths[v].emplace_back(0, -1); paths[v].emplace_back(0, -1); sort(paths[v].begin(), paths[v].end(), greater{}); paths[v].resize(3); longest[v].emplace_back(0, -1); sort(longest[v].begin(), longest[v].end(), greater{}); longest[v].resize(2); } void dfs2(int v, int p) { for (auto u : G[v]) if (u != p) { // 辺{u,v}を消す int u_len = max(longest[u][0].first, paths[u][0].first + paths[u][1].first); int i1 = u == paths[v][0].second ? 1 : 0; int i2 = u == paths[v][i1 + 1].second ? i1 + 2 : i1 + 1; int j = u == longest[v][0].second ? 1 : 0; int v_len = max(longest[v][j].first, paths[v][i1].first + paths[v][i2].first); ans = min(ans, max({u_len, v_len, (u_len + 1) / 2 + (v_len + 1) / 2 + 1})); pair<int, int> val1(-1, -1), val2(-1, -1); auto it1 = find_if(all(paths[v]), [&](auto&& a) { return a.second == u; }); if (it1 != paths[v].end()) val1 = *it1, paths[v].erase(it1); auto it2 = find_if(all(longest[v]), [&](auto&& a) { return a.second == u; }); if (it2 != longest[v].end()) val2 = *it2, longest[v].erase(it2); paths[u].emplace_back(paths[v][0].first + 1, v); longest[u].emplace_back( max(longest[v][0].first, paths[v][0].first + paths[v][1].first), v); sort(paths[u].begin(), paths[u].end(), greater{}); sort(longest[u].begin(), longest[u].end(), greater{}); dfs2(u, v); if (val1.first != -1) paths[v].insert(it1, val1); if (val2.first != -1) longest[v].insert(it2, val2); } } int main() { scanf("%d", &n); assert(2 <= n && n <= 100000); dsu dsu(n); rep(_, n-1) { int u, v; scanf("%d%d", &u, &v); assert(1 <= u && u <= n); assert(1 <= v && v <= n); assert(u != v); u--, v--; assert(dsu.unite(u, v)); G[u].push_back(v); G[v].push_back(u); } dfs1(0, -1); ans = INT_MAX; dfs2(0, -1); printf("%d\n", ans); }