1 条题解
-
0
自动搬运
来自洛谷,原作者为

奇米
**搬运于
2025-08-24 22:12:19,当前版本为作者最后更新于2020-09-13 22:57:19,作者可能在搬运后再次修改,您可在原文处查看最新版自动搬运只会搬运当前题目点赞数最高的题解,您可前往洛谷题解查看更多
以下是正文
题解 -
题目描述
给出一棵 个点的树,每个点上有一种颜色。现在请你求出对于每一种颜色,树上有多少条链包含该种颜色的所有点。
一道细节很多的题目。
首先我们很容易想到若一种颜色的数量 那么输出 。已经如果一种颜色在大于等于三棵子树中出现过那么输出 。
于是我们考虑 的情况,这个也比较容易答案就是这个节点为根的子树大小相互乘起来再累加。
难点在于某种颜色有两个端点的情况的处理。我们记 为颜色 到现在出现的个数,并且记录 表示在当前子树中有几个颜色为 的个数。那么每次递归完一颗子树看 是否改变,若改变那么 加一。若 则表示其为该颜色的某一个端点。那么答案即为两个端的子树大小乘积,具体实现看代码。
时间复杂度
const int N=3e6+5; int n,m,cnt,t[N],col[N],siz[N],Siz[N]; int gs[N],cs[N],head[N],Nod[N],Ans[N]; struct Node { int nex,to; }; Node e[N<<1]; inline void jia(int u,int v) { e[++cnt].nex=head[u]; head[u]=cnt; e[cnt].to=v; } inline void dfs(int u,int fa) { siz[u]=1; int Col=col[u],sum=0; int onend=0,Las=cs[Col]; GO(i,u) { int v=e[i].to; if(v==fa) continue; int las=cs[Col];//递归这颗子树之前的颜色个数 dfs(v,u); Siz[u]+=siz[u]*siz[v];//对就1个该颜色答案的统计 siz[u]+=siz[v]; if(las^cs[Col]) sum++,onend=v;//记录端点 } Siz[u]+=siz[u]*(n-siz[u]); if(Las||t[Col]-1!=cs[Col]) sum++;//子树u自己本身也要统计进去 cs[Col]++; if(sum==1) { if(!gs[Col]) Nod[Col]=u; else { int mul=(onend)?n-siz[onend]:siz[u]; Ans[Col]=siz[Nod[Col]]*mul; } gs[Col]++;//统计有几个子树有颜色Col } } signed main() { io.read(n); For(i,1,n) { io.read(col[i]); t[col[i]]++; Nod[col[i]]=i; } For(i,1,n-1) { int x,y; io.read(x),io.read(y); jia(x,y),jia(y,x); } dfs(1,0); For(i,1,n) { int ret=0; if(!t[i]) ret=n*(n-1)/2ll;//没有这种颜色 if(t[i]==1) ret=Siz[Nod[i]]; //有1种 if(gs[i]==2) ret=Ans[i];//两个端点情况 io.write(ret),puts(""); } return 0; }
- 1
信息
- ID
- 4596
- 时间
- 1000ms
- 内存
- 1024MiB
- 难度
- 5
- 标签
- 递交数
- 0
- 已通过
- 0
- 上传者