zoukankan      html  css  js  c++  java
  • 树的重心

    求重心的本质,其实就是枚举每一个节点的所有子树,使得该节点中的最大子树最小化

    #include <iostream>
    #include <cstring>
    using namespace std;
    const int N = 2e5 + 10;
    int e[N],h[N],ne[N],mds[N],ds[N],rt,idx,n;
    // mds代表所有子树中最大的子树,ds代表所有子树的总和,rt代表重心
    void add(int a,int b) {
        e[idx] = b;
        ne[idx] = h[a];
        h[a] = idx ++;
    }
    void dfs(int u,int fa) {
        mds[u] = 0;
        ds[u] = 1;
        for(int i = h[u]; ~i;i = ne[i]) {
            int j = e[i];
            if(j == fa) continue;
            dfs(j,u);
            ds[u] += ds[j];// 算出所有子树的总和
            mds[u] = max(mds[u],ds[j]);// 求出所有子树中最大的子树
        }
        mds[u] = max(mds[u], n - ds[u]);// 把父节点也看作一颗子树
        if(rt == 0 || mds[u] < mds[rt]) rt = u;
        if(mds[u] == mds[rt] && u < rt) rt = u;// 取编号较小的一个点
    }
    int main() {
        int t;
        cin >> t;
        while(t --) {
            memset(mds,0,sizeof mds);
            memset(ds,0,sizeof ds);
            memset(h,-1,sizeof h);
            rt = idx = 0;
            cin >> n;
            for(int i = 0;i < n - 1; ++i) {
                int a,b;
                cin >> a >> b;
                add(a,b);
                add(b,a);
            }
            dfs(1,0);
            cout << rt <<' ' << mds[rt] << endl;
        }
        return 0;
    }
    

    模板题

    Balancing Act

  • 相关阅读:
    大数板子
    数位dp
    BM算法
    牛客小白月赛7
    划分树
    可持久化线段树
    素数
    一些数学公式
    线性基
    ConcurrentHashMap 并发HashMap原理分析
  • 原文地址:https://www.cnblogs.com/lukelmouse/p/13171844.html
Copyright © 2011-2022 走看看