zoukankan      html  css  js  c++  java
  • CF1187E Tree Paiting(树形DP+换根)

    CF1187E Tree Painting(树形DP+换根)

    题意:

    给出一棵树,开始所有的点都是白色的,你可以给树上的点染色。

    一次染色,你可以选择任意一个和已经被染成黑色的点相邻的白点,将其染成黑色,并获得等同于这个点所在的白色连通块的点数的分数。询问最大分数。

    题解:

    先一遍DFS处理出每个点的子树节点数量,第二遍DFS不断更换根节点,维护当前所有子树节点数量和,取最大的就是答案。

    //第一步选择一个白色顶点涂成黑色
    //之后每步选择一个与黑色顶点相邻的白色顶点涂成黑色
    //每涂一次的分数是这个点所在的白色连通块的点数
    //询问最大分数
    #include<bits/stdc++.h>
    using namespace std;
    const int maxn=2e5+100;
    typedef long long ll;
    int n;vector<int> g[maxn];
    int size[maxn];
    void dfs (int u,int f) {
        size[u]=1;
        for (int v:g[u]) {
            if (v==f) continue;
            dfs(v,u);
            size[u]+=size[v];
        }
    }
    ll sum,ans=0;
    //换根
    //从u转移到v
    //size[u]变成n-size[v]
    //size[v]变成n
    void dfs1 (int u,int f) {
        for (int v:g[u]) {
            if (v==f) continue;
            int t1=size[u],t2=size[v];
            size[u]-=size[v];
            size[v]=n;
            sum+=size[u]-t1+size[v]-t2;
            ans=max(ans,sum);
            dfs1(v,u);
            sum-=size[u]-t1+size[v]-t2;
            size[u]=t1;size[v]=t2;
        }
    } 
    int main () {
        scanf("%d",&n);
        for (int i=1;i<n;i++) {
            int x,y;
            scanf("%d%d",&x,&y);
            g[x].push_back(y);
            g[y].push_back(x);
        }
        dfs(1,0);
        for (int i=1;i<=n;i++) sum+=size[i];
        ans=sum;
        dfs1(1,0);
        printf("%lld
    ",ans);
    } 
  • 相关阅读:
    获取当前时间并格式化,CTime类
    疑问:VS在调试的过程中,总是会提示正在加载picface.dll的符号,然后卡死在那
    Markup解析XML——文档,说明
    .net Core 获取当前程序路径
    Excel中的细节
    心血来潮尝试一个小项目(WinForm)
    bat文件以管理员运行
    DataGridView一些总结
    常见辅助类、方法
    向txt文件中添加或者追加文字字符串
  • 原文地址:https://www.cnblogs.com/zhanglichen/p/13719040.html
Copyright © 2011-2022 走看看