DSU on tree 学习笔记
DSU on tree 通常用来解决不带修树上子树问题。
主要思想:
- 剖分。
- 先搜轻儿子,记录轻儿子子树的答案,删去轻儿子的贡献。
- 搜重儿子,记录重儿子子树的答案,保留重儿子的贡献。
- 回溯,重新搜轻儿子,合并轻儿子子树的贡献,构成本子树的答案。
由于每一次合并答案都是把小的子树合并到大的子树里去,因此合并后子树的大小必定大于等于小子树的大小,所以小子树内元素个数至多被合并
#include<bits/stdc++.h>
using namespace std;
#define ll long long
const int N=1e5+10;
struct edge
{
int u,v,nxt;
}e[N<<1];
int head[N],tot;
void add(int u,int v)
{
e[++tot]=edge{u,v,head[u]};
head[u]=tot;
}
int n;
int col[N];
int siz[N],son[N];
void dfs1(int u,int f)
{
siz[u]=1;
for(int i=head[u];i;i=e[i].nxt)
{
int v=e[i].v;
if(v==f) continue;
dfs1(v,u);
siz[u]+=siz[v];
if(siz[v]>siz[son[u]]) son[u]=v;
}
}
int cnt[N];
ll ans[N];
int maxn;
ll sum;
void add(int u,int f,int s)
{
cnt[col[u]]++;
if(cnt[col[u]]>maxn) maxn=cnt[col[u]],sum=col[u];
else if(cnt[col[u]]==maxn) sum+=col[u];
for(int i=head[u];i;i=e[i].nxt)
{
int v=e[i].v;
if(v==f||v==s) continue;
add(v,u,s);
}
}
void sub(int u,int f)
{
cnt[col[u]]--;
for(int i=head[u];i;i=e[i].nxt)
{
int v=e[i].v;
if(v==f) continue;
sub(v,u);
}
}
void dfs2(int u,int f,int s)
{
for(int i=head[u];i;i=e[i].nxt)
{
int v=e[i].v;
if(v==f||v==son[u]) continue;
dfs2(v,u,0);
}
if(son[u]) dfs2(son[u],u,1);
add(u,f,son[u]);
ans[u]=sum;
if(!s) sub(u,f),sum=maxn=0;
}
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++) scanf("%d",&col[i]);
for(int i=1;i<n;i++)
{
int u,v;
scanf("%d%d",&u,&v);
add(u,v),add(v,u);
}
dfs1(1,0),dfs2(1,0,0);
for(int i=1;i<=n;i++) printf("%lld ",ans[i]);
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?