結果

問題 No.1976 Cut then Connect
ユーザー shino16shino16
提出日時 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
権限があれば一括ダウンロードができます

ソースコード

diff #

// 制約チェック

#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);
}

0