結果
| 問題 | No.922 東北きりきざむたん | 
| コンテスト | |
| ユーザー |  | 
| 提出日時 | 2019-11-08 22:52:27 | 
| 言語 | C++17 (gcc 13.3.0 + boost 1.87.0) | 
| 結果 | 
                                WA
                                 
                             | 
| 実行時間 | - | 
| コード長 | 7,581 bytes | 
| コンパイル時間 | 3,143 ms | 
| コンパイル使用メモリ | 234,264 KB | 
| 最終ジャッジ日時 | 2025-01-08 02:41:30 | 
| ジャッジサーバーID (参考情報) | judge1 / judge1 | 
(要ログイン)
| ファイルパターン | 結果 | 
|---|---|
| sample | AC * 4 | 
| other | AC * 8 WA * 18 | 
ソースコード
#include <bits/stdc++.h>
#define LLI long long int
#define FOR(v, a, b) for(LLI v = (a); v < (b); ++v)
#define FORE(v, a, b) for(LLI v = (a); v <= (b); ++v)
#define REP(v, n) FOR(v, 0, n)
#define REPE(v, n) FORE(v, 0, n)
#define REV(v, a, b) for(LLI v = (a); v >= (b); --v)
#define ALL(x) (x).begin(), (x).end()
#define RALL(x) (x).rbegin(), (x).rend()
#define ITR(it, c) for(auto it = (c).begin(); it != (c).end(); ++it)
#define RITR(it, c) for(auto it = (c).rbegin(); it != (c).rend(); ++it)
#define EXIST(c,x) ((c).find(x) != (c).end())
#define fst first
#define snd second
#define popcount __builtin_popcount
#define UNIQ(v) (v).erase(unique(ALL(v)), (v).end())
#define bit(i) (1LL<<(i))
#ifdef DEBUG
#include <misc/C++/Debug.cpp>
#else
#define dump(...) ((void)0)
#endif
#define gcd __gcd
using namespace std;
template <class T> constexpr T lcm(T m, T n){return m/gcd(m,n)*n;}
template <typename I> void join(ostream &ost, I s, I t, string d=" "){for(auto i=s; i!=t; ++i){if(i!=s)ost<<d; ost<<*i;}ost<<endl;}
template <typename T> istream& operator>>(istream &is, vector<T> &v){for(auto &a : v) is >> a; return is;}
template <typename T, typename U> bool chmin(T &a, const U &b){return (a>b ? a=b, true : false);}
template <typename T, typename U> bool chmax(T &a, const U &b){return (a<b ? a=b, true : false);}
template <typename T, size_t N, typename U> void fill_array(T (&a)[N], const U &v){fill((U*)a, (U*)(a+N), v);}
struct Init{
  Init(){
    cin.tie(0);
    ios::sync_with_stdio(false);
    cout << fixed << setprecision(12);
    cerr << fixed << setprecision(12);
  }
}init;
template <typename Cost = int> class Edge{
public:
  int from,to;
  Cost cost;
  Edge() {}
  Edge(int to, Cost cost): to(to), cost(cost){}
  Edge(int from, int to, Cost cost): from(from), to(to), cost(cost){}
  Edge rev() const {return Edge(to,from,cost);}
  
  friend ostream& operator<<(ostream &os, const Edge &e){
    os << "(FROM: " << e.from << "," << "TO: " << e.to << "," << "COST: " << e.cost << ")";
    return os;
  }
};
template <typename T> using Graph = vector<vector<Edge<T>>>;
template <typename T> using Tree = vector<vector<Edge<T>>>;
template <typename C, typename T> void add_edge(C &g, int from, int to, T w){
  g[from].push_back(Edge<T>(from, to, w));  
}
template <typename C, typename T> void add_undirected(C &g, int a, int b, T w){
  g[a].push_back(Edge<T>(a, b, w));
  g[b].push_back(Edge<T>(b, a, w));
}
class UnionFind{
  vector<int> parent, depth, size;
  int count;
public:
  UnionFind(int n): parent(n), depth(n,1), size(n,1), count(n){
    iota(ALL(parent),0);
  }
  int get_root(int i){
    if(parent[i] == i) return i;
    else return parent[i] = get_root(parent[i]);
  }
  bool is_same(int i, int j){return get_root(i) == get_root(j);}
  int merge(int i, int j){
    int ri = get_root(i), rj = get_root(j);
    if(ri == rj) return ri;
    else{
      --count;
      if(depth[ri] < depth[rj]){
        parent[ri] = rj;
        size[rj] += size[ri];
        return rj;
      }else{
        parent[rj] = ri;
        size[ri] += size[rj];
        if(depth[ri] == depth[rj]) ++depth[ri];
        return ri;
      }
    }
  }
  int get_size(int i){return size[get_root(i)];}
  int count_group(){return count;}
};
template <typename T> class LCA{
private:
  vector<vector<int>> parent;
  int n, log2n;
  
  void dfs(const Tree<T> &tree, int cur, int par, int d){
    parent[cur][0] = par;
    depth[cur] = d;
    for(auto &e : tree[cur]){
      if(e.to != par){
        dist[e.to] = dist[cur] + e.cost;
        dfs(tree, e.to, cur, d+1);
      }
    }
  }
  
public:
  vector<int> depth;
  vector<T> dist;
  LCA(){}
  
  LCA(const Tree<T> &tree, int root):
    n(tree.size()), depth(n), dist(n)
  {
    log2n = (int)ceil(log(n) / log(2)) + 1;
    parent = vector<vector<int>>(n, vector<int>(log2n, 0));
    dfs(tree, root, -1, 0);
    REP(k,log2n-1){
      REP(v,n){
        if(parent[v][k] == -1) parent[v][k+1] = -1;
        else parent[v][k+1] = parent[parent[v][k]][k];
      }
    }
  }
  int query(int a, int b){
    if(depth[a] >= depth[b]) swap(a,b);
    REP(k,log2n) if((depth[b] - depth[a]) >> k & 1) b = parent[b][k];
    if(a == b) return a;
    REV(k, log2n-1, 0) if(parent[a][k] != parent[b][k]){a = parent[a][k]; b = parent[b][k];}
    return parent[a][0];
  }
  T distance(int a, int b){
    return dist[a] + dist[b] - 2 * dist[query(a,b)];
  }
};
template <typename F>
struct FixPoint : F{
  explicit constexpr FixPoint(F &&f) noexcept : F(forward<F>(f)){}
  template <typename... Args>
  constexpr decltype(auto) operator()(Args &&... args) const {
    return F::operator()(*this, forward<Args>(args)...);
  }
};
template <typename F>
static inline constexpr decltype(auto) make_fix_point(F &&f){
  return FixPoint<F>(forward<F>(f));
}
int main(){
  int N,M,Q;
  while(cin >> N >> M >> Q){
    Graph<int> g(N);
    UnionFind uf(N);
    
    REP(i,M){
      int u, v; cin >> u >> v;
      --u, --v;
      add_undirected(g, u, v, 1);
      uf.merge(u, v);
    }
    LLI ans = 0;
    
    vector<vector<int>> vertex(N);
    map<int, pair<int,int>> new_node;
    vector<Tree<int>> trees(N);
    vector<LCA<int>> lcas(N);
    REP(i,N) if(uf.get_root(i) == i) trees[i] = Tree<int>(uf.get_size(i));
    
    REP(i,N){
      int k = uf.get_root(i);
      vertex[k].push_back(i);
      new_node[i] = make_pair(k, vertex[k].size()-1);
    }
    REP(i,N){
      for(auto &e : g[i]){
        int k = uf.get_root(e.from);
        add_edge(trees[k], new_node[e.from].snd, new_node[e.to].snd, 1);
      }
    }
    REP(i,N){
      if(trees[i].size()) lcas[i] = LCA(trees[i], 0);
    }
    //dump(trees);
    vector<vector<bool>> plans(N);
    REP(i,N){
      plans[i] = vector<bool>(trees[i].size());
    }
    REP(i,Q){
      int a,b; cin >> a >> b;
      --a, --b;
      if(uf.is_same(a, b)){
        ans += lcas[uf.get_root(a)].distance(new_node[a].snd, new_node[b].snd);
      }else{
        plans[new_node[a].fst][new_node[a].snd] = true;
        plans[new_node[b].fst][new_node[b].snd] = true;
      }
    }
    dump(ans);
    REP(i,N){
      if(trees[i].size() == 0) continue;
      int L = trees[i].size();
      auto &tree = trees[i];
      int s = 0;
      REP(j,L) if(plans[i][j]) ++s;
      dump(plans[i]);
      dump(tree);
      vector<int> count(L);
      auto dfs1 =
        make_fix_point([&](auto &&f, int cur, int par) -> LLI{
                         LLI ret = 0;
                         for(auto &e : tree[cur]){
                           if(e.to == par) continue;
                           ret += f(e.to, cur);
                           //if(plans[i][e.to]) ret += 1;
                           count[cur] += count[e.to];
                         }
                         ret += count[cur];
                         if(plans[i][cur]) count[cur] = 1;
                         return ret;
                       });
      LLI dist = dfs1(0, -1);
      dump(dist, s, count);
      LLI t = dist;
      auto dfs2 =
        make_fix_point([&](auto &&f, int cur, int par, LLI d) -> void{
                         for(auto &e : tree[cur]){
                           if(e.to == par) continue;
                           LLI nd = d - count[e.to] + (s - count[e.to]);
                           chmin(t, nd);
                           //dump(nd);
                           
                           f(e.to, cur, nd);
                         }
                       });
      dfs2(0, -1, dist);
      dump(t);
      ans += t;
    }
    cout << ans << endl;
  }
  return 0;
}
            
            
            
        