这题是个很好的点分治题目。
首先明确我们的目的,既然使用了点分治,那么就需要O(n)的时间内处理出重心rt以及“rt的儿子”的答案。
考虑重心的一个子树的根节点,如果这个点v的颜色是第一次出现,那么它对其他不在该子树的点的答案的贡献(包括rt)就是size[v]
然后我们再考虑这个子树,注意刚刚我们考虑的是子树的根节点,现在考虑的是整个子树对其他“rt儿子”的贡献。假设该子树的根节点是v,从v~rt上有num种颜色,那么对其他节点的贡献就是num*(size[rt]-size[v])
整体的步骤:
这就是全部了具体操作看代码比较好
#include
using namespace std;
#define ll long long
const int N = 1e5+10;
int head[N],tot,n;
int rt,sum,maxp[N],siz[N],cnt[N],V[N],color[N],much,maxx;
ll ans[N],num;
bool vis[N];
struct edge{
int v,nex,w;
edge(){}
edge(int v, int nex, int w) : v(v), nex(nex), w(w) {}
}edges[N<<1];
void add(int u,int v,int w){
edges[tot]=edge(v,head[u],w);
head[u]=tot++;
}
void dfs1(int u,int fa){
siz[u]=1;
cnt[V[u]]++;
for(int i = head[u];~i;i=edges[i].nex){
int v = edges[i].v;
if(vis[v]||v==fa)continue;
dfs1(v,u);
siz[u]+=siz[v];
}
//首次出现
if(cnt[V[u]]==1){
sum+=siz[u];
color[V[u]]+=siz[u];
}
cnt[V[u]]--;
}
//找重心
void getrt(int u,int fa){
siz[u]=1,maxp[u]=0;
for(int i=head[u];~i;i=edges[i].nex){
int v =edges[i].v;
if(vis[v]||v==fa)continue;
getrt(v,u);
siz[u]+=siz[v];
if(maxp[u]<siz[v])maxp[u]=siz[v];
}
maxp[u]=max(maxp[u],maxx-siz[u]);
if(maxp[u]<maxp[rt])rt=u;
}
void change(int u,int fa,int value){
cnt[V[u]]++;
for(int i=head[u];~i;i=edges[i].nex){
int v = edges[i].v;
if(v==fa||vis[v])continue;
change(v,u,value);
}
if(cnt[V[u]]==1){
sum += 1ll*siz[u]*value;
color[V[u]]+=1ll*siz[u]*value;
}
cnt[V[u]]--;
}
void dfs2(int u,int fa){
cnt[V[u]]++;
// 从u~p上的颜色数统计
if(cnt[V[u]]==1){
sum-=color[V[u]];
num++;
}
ans[u]+=sum+num*much;
for(int i=head[u];~i;i=edges[i].nex){
int v = edges[i].v;
if(vis[v]||v==fa)continue;
dfs2(v,u);
}
// 第一次出现的
if(cnt[V[u]]==1){
sum+=color[V[u]];
num--;
}
cnt[V[u]]--;
}
void clear(int u,int fa){
cnt[V[u]]=0;
color[V[u]]=0;
for(int i=head[u];~i;i=edges[i].nex){
int v = edges[i].v;
if(vis[v]||v==fa)continue;
clear(v,u);
}
}
void solve(int u,int fa){
dfs1(u,fa);
ans[u]+=sum-color[V[u]]+siz[u];
for(int i=head[u];~i;i=edges[i].nex){
int v = edges[i].v;
if(vis[v]||v==fa)continue;
// 清除子树的贡献,因为计算的是经过根节点的路径
// 而我们计算当前子节点的时候需要清除点分治的第二种情况
cnt[V[u]]++;
sum-=siz[v];
color[V[u]]-=siz[v];
change(v,u,-1);
cnt[V[u]]--;
much=siz[u]-siz[v];
// 计算子树的贡献
dfs2(v,u);
// 计算完了之后再加上当前子树的贡献
cnt[V[u]]++;
sum+=siz[v];
color[V[u]]+=siz[v];
change(v,u,1);
cnt[V[u]]--;
}
sum = 0,num = 0;
// 删除当前点
clear(u,fa);
}
//点分治
void divide(int u,int fa){
vis[u]=1;
solve(u,fa);
for(int i=head[u];~i;i=edges[i].nex){
int v = edges[i].v;
if(vis[v])continue;
maxp[rt=0]=maxx=siz[v];
getrt(v,0);
getrt(rt,0);
divide(rt,0);
}
}
int main(){
memset(head,-1,sizeof(head));
scanf("%d",&n);
for(int i=1;i<=n;i++)scanf("%d",&V[i]);
for(int i=0;i<n-1;i++){
int u,v,w;
scanf("%d %d",&u,&v);
add(u,v,0);
add(v,u,0);
}
maxp[0]=maxx=n;
getrt(1,0);
getrt(rt,0);
divide(rt,0);
for(int i=1;i<=n;i++)printf("%lld\n",ans[i]);
}