zoukankan      html  css  js  c++  java
  • 600E

    题:https://codeforces.com/problemset/problem/600/E

    题意:一棵树有n个结点,每个结点都是一种颜色,每个颜色有一个编号,求树中每个子树的最多的颜色编号的和,对于每个结点都输出答案。

    分析:考虑暴力算法,对于每个节点只是清空计数数组,再对其子树颜色进行统计,复杂度o(n^2);

       接着我们发现最后一个子树的清空是必要的,所以我们把重儿子当作这个子树,就可以让复杂度降为o(nlogn)级别;

    #include<bits/stdc++.h>
    using namespace std;
    #define pb push_back
    typedef long long ll;
    const int M=2e5+5;
    ll ans[M];
    ll countt[M],sz[M],son[M],col[M],vis[M];
    vector<int>g[M];
    ll nownum,maxx;
    void dfs1(int u,int fa){
        sz[u]=1;
        for(int i=0;i<g[u].size();i++){
            int v=g[u][i];
            if(v!=fa){
                dfs1(v,u);
                sz[u]+=sz[v];
                if(sz[v]>sz[son[u]])
                    son[u]=v;
            }
        }
    }
    
    void update(int u,int k,int fa){//k的取值为1和-1,分别对应累加和清除
        countt[col[u]]+=k;
        if(maxx<countt[col[u]])
            maxx=countt[col[u]],nownum=col[u];
        else if(maxx==countt[col[u]])
            nownum+=col[u];
        for(int i=0;i<g[u].size();i++){
            int v=g[u][i];
            if(v!=fa&&!vis[v])
                update(v,k,u);
        }
    }
    void dfs2(int u,int fa,int sign){
        //cout<<u<<endl;
        for(int i=0;i<g[u].size();i++){
            int v=g[u][i];
            if(v!=fa&&son[u]!=v)
                dfs2(v,u,0);
        }
        if(son[u])//再访问重儿子,一定要打上vis标记
            dfs2(son[u],u,1),vis[son[u]]=1;
        update(u,1,fa);
        ans[u]=nownum;
        if(son[u])
            vis[son[u]]=0;
        if(!sign)///轻儿子就消除影响 
            update(u,-1,fa),nownum=maxx=0;
        
    }
    int main(){
        int n;
        scanf("%d",&n);
        for(int i=1;i<=n;i++)
            scanf("%d",&col[i]);
        for(int u,v,i=1;i<n;i++){
            scanf("%d%d",&u,&v);
            g[u].pb(v);
            g[v].pb(u);
        }
        dfs1(1,0);
        
        dfs2(1,0,0);
        for(int i=1;i<=n;i++)
            printf("%I64d ",ans[i]);
        return 0;
    }
    View Code
  • 相关阅读:
    hduoj 1865 1string 【大数】【菲波那切数列】
    poj 1664 放苹果【M的N划分】
    新年第一篇
    3、XCode: 如何添加自定义代码片段
    2、文件夹
    1、获取当前屏幕显示的页面
    运算符
    表单数据接收
    PHP进入MySQL数据库
    my SQL认识和进入
  • 原文地址:https://www.cnblogs.com/starve/p/12233144.html
Copyright © 2011-2022 走看看