CF600ELomset gelral 题解(树上启发式合并)

Posted invictus-ocean

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了CF600ELomset gelral 题解(树上启发式合并)相关的知识,希望对你有一定的参考价值。

题目链接

题目大意:给出一颗含有$n$个结点的树,每个节点有一个颜色。求树中每个子树最多的颜色的编号和。

-------------------------

树上启发式合并(dsu on tree)。

我们先考虑暴力怎么做。遍历整颗树,暴力枚举子树然后用桶维护颜色个数。这样做是$O(n^2)$的,显然会T。我们需要一种更快的算法:树上启发式合并。

关于启发式算法的介绍,详见OI Wiki。本文只介绍树上启发式合并算法。本题的解法:

每处理完一颗子树,我们都要把桶清空一次,以免对它的兄弟造成影响。而这样做还要从它的祖先遍历一遍,浪费时间。

我们发现:遍历最后一颗子树时,桶是不用清空的。因为遍历完那颗子树后可以直接把答案加入$ans$中。那我们肯定选重儿子啊,省时省力。遍历轻儿子相对不费事。

看起来是不是没有快多少?实际上它是$O(nlog n)$的。下面是证明:

对于每个节点,它被计算的次数就是它到根节点路径的轻边个数。

而结点往上跳一次,子树大小至少为原来两倍,所以轻边个数最多是$log n$。所以时间复杂度$O(nlog n)$。

证明过程跟树链剖分的有点像。

代码:

#include<bits/stdc++.h>
#define int long long
using namespace std;
int n,color[200005],bucket[200005],ans[200005];
int size[200005],son[200005],sum,mx;
int head[200005],cnt;
struct node
{
    int next,to;
}edge[200005];
inline int read()
{
    int x=0,f=1;char ch=getchar();
    while(!isdigit(ch)){if (ch==-) f=-1;ch=getchar();}
    while(isdigit(ch)){x=x*10+ch-0;ch=getchar();}
    return x*f;
}
inline void add(int from,int to)
{
    edge[++cnt].next=head[from];
    edge[cnt].to=to;
    head[from]=cnt;
}
inline void dfs_son(int now,int fa)
{
    size[now]=1;
    int mx=0,p=0;
    for (int i=head[now];i;i=edge[i].next)
    {
        int to=edge[i].to;
        if (to==fa) continue;
        dfs_son(to,now);
        size[now]+=size[to];
        if (size[to]>mx)
        {
            mx=size[to];
            p=to;
        }
    }
    if (p) son[p]=1;
}
void getans(int x,int f,int p){
    bucket[color[x]]++;
    if(bucket[color[x]]>mx){
        mx=bucket[color[x]];
        sum=color[x];
    }else if(bucket[color[x]]==mx)sum+=color[x];
    for(int i=head[x];i;i=edge[i].next){
        int y=edge[i].to;
        if(y==f || y==p)continue;
        getans(y,x,p);
    }
}
inline void init(int now,int fa)
{
    bucket[color[now]]--;
    for (int i=head[now];i;i=edge[i].next)
    {
        int to=edge[i].to;
        if (to==fa) continue;
        init(to,now);
    }
}
inline void dfs(int now,int fa)
{
    int p=0;
    for (int i=head[now];i;i=edge[i].next)
    {
        int to=edge[i].to;
        if (to==fa) continue;
        if (!son[to])
        {
            dfs(to,now);
            init(to,now);
            sum=mx=0;
        }
        else p=to;
    }
    if (p) dfs(p,now);
    getans(now,fa,p);
    ans[now]=sum;
}
signed main()
{
    n=read();
    for (int i=1;i<=n;i++) color[i]=read();
    for (int i=1;i<n;i++)
    {
        int x=read(),y=read();
        add(x,y);add(y,x);
    }
    dfs_son(1,0);
    dfs(1,0);
    for (int i=1;i<=n;i++) printf("%lld ",ans[i]);
    return 0;
}

 

以上是关于CF600ELomset gelral 题解(树上启发式合并)的主要内容,如果未能解决你的问题,请参考以下文章

CF600E Lomsat gelral(树上启发式合并)

CF 600 E Lomsat gelral —— 树上启发式合并

CF600E Lomsat gelral(树上启发式合并)

cf600E. Lomsat gelral(树上启发式合并)

CF600ELomsat gelral——树上启发式合并

题解Lomsat gelral [CF600E]