P2971 [USACO10HOL]Cow Politics G
题意
一个树上每一个点都有一个组别,求相同组别的点对相差的最大距离。
分析
首先对于任意一个组别,深度最大的点一定在答案的点对里。
证明
假设答案的点对里没有深度最大的点,设深度最大的点为
,设点对中的点为 ,假设 , 为 的 。
1.若
属于 的不同子树,因为 , ,若 不在 的子树内,那么,把 换为 会更优。若 在 的子树内,如果 ,显然可以将 换成 。如果 ,那么很明显,将 换成 一定比点对 的答案更优,或者点对 比点对 更优,显然不论哪种情况, 都在答案里。
2.若
属于 的同一棵子树,即 ,因为 , ,若 不在 的子树内,那么把 换成 会更优,若在 子树内,那么显然把 换成 更优。
所以,对于不同组别,我们只需要求出深度最大的点即可,在
#include<bits/stdc++.h> #define N 200005 using namespace std; int read(){ int x=0,f=1;char ch=getchar(); while(ch<'0'||ch>'9'){if(ch=='-')f=-f;ch=getchar();} while(ch>='0'&&ch<='9'){x=(x<<1)+(x<<3)+(ch^48);ch=getchar();} return x*f; } queue<int> q; int n,k,root,tot,t; int a[N],p[N],d[N],f[N][25],maxd[N],ans[N],pos[N]; int Head[N],to[N<<1],Next[N<<1]; void add(int u,int v){to[++tot]=v,Next[tot]=Head[u],Head[u]=tot;} void bfs(int s){ q.push(s);d[s]=1,maxd[a[s]]=1; while(!q.empty()){ int x=q.front();q.pop(); for(int i=Head[x];i;i=Next[i]){ int y=to[i];if(d[y]) continue; d[y]=d[x]+1,f[y][0]=x; if(maxd[a[y]]<d[y]) pos[a[y]]=y; maxd[a[y]]=max(maxd[a[y]],d[y]); for(int j=1;j<=t;j++) f[y][j]=f[f[y][j-1]][j-1]; q.push(y); } } } int lca(int x,int y){ if(d[x]>d[y]) swap(x,y); for(int j=t;j>=0;--j) if(d[f[y][j]]>=d[x]) y=f[y][j]; if(x==y) return x; for(int j=t;j>=0;--j) if(f[y][j]!=f[x][j]) x=f[x][j],y=f[y][j]; return f[x][0]; } int DIS(int x,int y){return d[x]+d[y]-2*d[lca(x,y)];} signed main(){ n=read(),k=read(); for(int i=1;i<=n;++i){ a[i]=read(),p[i]=read(); if(p[i]==0) root=i; else add(i,p[i]),add(p[i],i); } t=log(n)/log(2)+1; bfs(root); for(int i=1;i<=n;++i) ans[a[i]]=max(ans[a[i]],DIS(pos[a[i]],i)); for(int i=1;i<=k;++i) printf("%d\n",ans[i]); return 0; }
本文作者:Aurora-JC
本文链接:https://www.cnblogs.com/jiangchen4122/p/16840927.html
版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步