P3884 [JLOI2009]二叉树问题
Posted for-miku
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了P3884 [JLOI2009]二叉树问题相关的知识,希望对你有一定的参考价值。
---------------------
链接:Miku
---------------------
这一道题只需要在倍增lca的板子上改一改就可以了。
宽度和深度可以在倍增lca的dfs预处理的时候判断一下就可以,至于最后问的两点之间的距离
首先需要求出两点公共祖先的位置,然后计算他们深度的差,并且按照题目要求分别处理即可
--------------------
#include<iostream> #include<cstdio> using namespace std; int head[2*500001]; int p; int md; int mk; int wei[10000]; struct b{ int to; int ne; } e[2*500001]; int fa[2*500001][30]; int dep[2*500001]; int n,m,s; int logg[2*500001]; int x,y; void dfs(int now,int fat){ dep[now]=dep[fat]+1; md=max(md,dep[now]); wei[dep[now]]++; mk=max(mk,wei[dep[now]]); fa[now][0]=fat; for(int i=1;(1<<i)<=dep[now];++i){ fa[now][i]=fa[fa[now][i-1]][i-1]; } for(int i=head[now];i;i=e[i].ne){ if(e[i].to!=fat) dfs(e[i].to,now); } } int lca(int x,int y){ int ans=0; if(dep[x]>dep[y]){ int v=dep[y]; ans+=(2*(dep[x]-dep[y])); while(dep[x]>dep[y]){ x=fa[x][logg[dep[x]-dep[y]]-1]; } if(x==y) return ans; for(int k=logg[dep[x]]-1;k>=0;k--){ if(fa[x][k]!=fa[y][k]){ x=fa[x][k]; y=fa[y][k]; } } ans+=(3*(v-dep[fa[x][0]])); return ans; } else{ ans+=(dep[y]-dep[x]); int v=dep[x]; while(dep[x]<dep[y]){ y=fa[y][logg[dep[y]-dep[x]]-1]; } if(x==y) return ans; for(int k=logg[dep[x]]-1;k>=0;k--){ if(fa[x][k]!=fa[y][k]){ x=fa[x][k]; y=fa[y][k]; } } ans+=(3*(v-dep[fa[x][0]])); return ans; } } void add(int f,int t){ p++; e[p].to=t; e[p].ne=head[f]; head[f]=p; } int main(){ scanf("%d",&n); for(int i=1;i<=n-1;++i){ scanf("%d%d",&x,&y); add(x,y); add(y,x); } dfs(1,0); cout<<md<<endl<<mk<<endl; for(int i=1;i<=n;++i) logg[i]=logg[i-1]+(1<<logg[i-1]==i); for(int i=1;i<=1;++i){ scanf("%d%d",&x,&y); cout<<lca(x,y)<<endl; } return 0; }
以上是关于P3884 [JLOI2009]二叉树问题的主要内容,如果未能解决你的问题,请参考以下文章
NC41 最长无重复子数组/NC133链表的奇偶重排/NC116把数字翻译成字符串/NC135 股票交易的最大收益/NC126换钱的最少货币数/NC45实现二叉树先序,中序和后序遍历(递归)(代码片段