lca 最近公共祖先

http://poj.org/problem?id=1330

  1 #include<cstdio>
  2 #include<cstring>
  3 #include<algorithm>
  4 #define mt(a,b) memset(a,b,sizeof(a))
  5 using namespace std;
  6 const int inf=0x3f3f3f3f;
  7 class LCA { ///最近公共祖先 build_o(n*logn) query_o(1)
  8     typedef int typec;///边权的类型
  9     static const int ME=2e4+10;///边的个数
 10     static const int MV=1e4+10;///点的个数
 11     static const int MR=MV<<1;///rmq的个数
 12     struct E {
 13         int v,next;
 14         typec w;
 15     } e[ME];
 16     int le,head[MV],H[MV],dex[MR],aa[MR],bb[MR],raa[MR][20],rbb[MR][20],Index;
 17     typec dist[MV];
 18     void init_dex() {
 19         dex[0]=-1;
 20         for(int i=1; i<MR; i++) {
 21             dex[i]=dex[i>>1]+1;
 22         }
 23     }
 24     void dfs(int u,int fa,int level) {
 25         aa[Index]=u;
 26         bb[Index++]=level;
 27         for(int i=head[u]; ~i; i=e[i].next) {
 28             int v=e[i].v;
 29             if(v==fa) continue;
 30             dist[v]=dist[u]+e[i].w;
 31             dfs(v,u,level+1);
 32             aa[Index]=u;
 33             bb[Index++]=level;
 34         }
 35     }
 36 public:
 37     LCA() { init_dex();};
 38     void init() {
 39         le=Index=0;
 40         mt(head,-1);
 41     }
 42     void add(int u,int v,typec w) {
 43         e[le].v=v;
 44         e[le].w=w;
 45         e[le].next=head[u];
 46         head[u]=le++;
 47     }
 48     void build(int root) {
 49         dist[root]=0;
 50         dfs(root,-1,0);
 51         mt(H,-1);
 52         mt(rbb,0x3f);
 53         for(int i=0; i<Index; i++) {
 54             int tmp=aa[i];
 55             if(H[tmp]==-1) H[tmp]=i;
 56             rbb[i][0]=bb[i];
 57             raa[i][0]=aa[i];
 58         }
 59         for(int i=1; (1<<i)<=Index; i++) {
 60             for(int j=0; j+(1<<i)<Index; j++) {
 61                 int next=j+(1<<(i-1));
 62                 if(rbb[j][i-1]<=rbb[next][i-1]) {
 63                     rbb[j][i]=rbb[j][i-1];
 64                     raa[j][i]=raa[j][i-1];
 65                 } else {
 66                     rbb[j][i]=rbb[next][i-1];
 67                     raa[j][i]=raa[next][i-1];
 68                 }
 69             }
 70         }
 71     }
 72     int query(int l,int r) {
 73         l=H[l];
 74         r=H[r];
 75         if(l>r) swap(l,r);
 76         int p=dex[r-l+1];
 77         r-=(1<<p)-1;
 78         return rbb[l][p]<=rbb[r][p]?raa[l][p]:raa[r][p];
 79     }
 80     typec getdist(int l,int r) {
 81         return dist[l]+dist[r]-2*dist[query(l,r)];
 82     }
 83 } gx;
 84
 85 const int M=1e4+10;
 86 int du[M];
 87 int main() {
 88     int t,n,u,v;
 89     while(~scanf("%d",&t)) {
 90         while(t--) {
 91             scanf("%d",&n);
 92             gx.init();
 93             mt(du,0);
 94             for(int i=1; i<n; i++) {
 95                 scanf("%d%d",&u,&v);
 96                 du[v]++;
 97                 gx.add(u,v,1);
 98                 gx.add(v,u,1);
 99             }
100             int rt=1;
101             for(int i=1; i<=n; i++) {
102                 if(du[i]==0) {
103                     rt=i;
104                     break;
105                 }
106             }
107             gx.build(rt);
108             scanf("%d%d",&u,&v);
109             printf("%d\n",gx.query(u,v));
110         }
111     }
112     return 0;
113 }

end

时间: 2024-07-29 08:47:07

lca 最近公共祖先的相关文章

Codeforces Round #362 (Div. 2) C. Lorenzo Von Matterhorn LCA(最近公共祖先)

C. Lorenzo Von Matterhorn time limit per test 1 second memory limit per test 256 megabytes input standard input output standard output Barney lives in NYC. NYC has infinite number of intersections numbered with positive integers starting from 1. Ther

POJ 1330 LCA最近公共祖先 离线tarjan算法

题意要求一棵树上,两个点的最近公共祖先 即LCA 现学了一下LCA-Tarjan算法,还挺好理解的,这是个离线的算法,先把询问存贮起来,在一遍dfs过程中,找到了对应的询问点,即可输出 原理用了并查集和dfs染色,先dfs到底层开始往上回溯,边并查集合并 一边染色,这样只要询问的两个点均被染色了,就可以输出当前并查集的最高父亲一定是LCA,因为我是从底层层层往上DSU和染色的,要么没被染色,被染色之后,肯定就是当前节点是最近的 #include <iostream> #include <

笔记:LCA最近公共祖先 Tarjan(离线)算法

LCA最近公共祖先 Tarjan他贱(离线)算法的基本思路及其算法实现 本文是网络资料整理或部分转载或部分原创,参考文章如下: https://www.cnblogs.com/JVxie/p/4854719.html http://blog.csdn.net/ywcpig/article/details/52336496 https://baike.baidu.com/item/最近公共祖先/8918834?fr=aladdin 最近公共祖先简称LCA(Lowest Common Ancesto

连通分量模板:tarjan: 求割点 &amp;&amp; 桥 &amp;&amp; 缩点 &amp;&amp; 强连通分量 &amp;&amp; 双连通分量 &amp;&amp; LCA(最近公共祖先)

PS:摘自一不知名的来自大神. 1.割点:若删掉某点后,原连通图分裂为多个子图,则称该点为割点. 2.割点集合:在一个无向连通图中,如果有一个顶点集合,删除这个顶点集合,以及这个集合中所有顶点相关联的边以后,原图变成多个连通块,就称这个点集为割点集合. 3.点连通度:最小割点集合中的顶点数. 4.割边(桥):删掉它之后,图必然会分裂为两个或两个以上的子图. 5.割边集合:如果有一个边集合,删除这个边集合以后,原图变成多个连通块,就称这个点集为割边集合. 6.边连通度:一个图的边连通度的定义为,最

[转]LCA 最近公共祖先

原文传送门orzJVxie Tarjan(离线)算法的基本思路及其算法实现 首先是最近公共祖先的概念(什么是最近公共祖先?): 在一棵没有环的树上,每个节点肯定有其父亲节点和祖先节点,而最近公共祖先,就是两个节点在这棵树上深度最大的公共的祖先节点. 换句话说,就是两个点在这棵树上距离最近的公共祖先节点. 所以LCA主要是用来处理当两个点仅有唯一一条确定的最短路径时的路径. 有人可能会问:那他本身或者其父亲节点是否可以作为祖先节点呢? 答案是肯定的,很简单,按照人的亲戚观念来说,你的父亲也是你的祖

(转)Tarjan应用:求割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)

本文转载自:http://hi.baidu.com/lydrainbowcat/item/f8a5ac223e092b52c28d591c 作者提示:在阅读本文之前,请确保您已经理解并掌握了基本的Tarjan算法,不会的请到http://hi.baidu.com/lydrainbowcat/blog/item/42a6862489c98820c89559f3.html阅读.   基本概念:   1.割点:若删掉某点后,原连通图分裂为多个子图,则称该点为割点. 2.割点集合:在一个无向连通图中,如

poj1330 lca 最近公共祖先问题学习笔记

首先推荐两个博客网址: http://dongxicheng.org/structure/lca-rmq/ http://scturtle.is-programmer.com/posts/30055.html [转]tarjan算法的步骤是(当dfs到节点u时): 1 在并查集中建立仅有u的集合,设置该集合的祖先为u 1 对u的每个孩子v:    1.1 tarjan之    1.2 合并v到父节点u的集合,确保集合的祖先是u 2 设置u为已遍历 3 处理关于u的查询,若查询(u,v)中的v已遍

LCA最近公共祖先

Nearest Common Ancestors Description A rooted tree is a well-known data structure in computer science and engineering. An example is shown below: In the figure, each node is labeled with an integer from {1, 2,...,16}. Node 8 is the root of the tree.

LCA 最近公共祖先 小结

以poj 1330为例,对LCA的3种常用的算法进行介绍,分别为 1. 离线tajian 2. 基于倍增法的LCA 3. 基于RMQ的LCA 1. 离线tajian /*poj 1330 Nearest Common Ancestors 题意: 给出一棵大小为n的树和一个询问(u,v), 问(u,v)的最近公共祖先. 限制: 2 <= n <= 10000 思路: 离线tajian */ #include<iostream> #include<cstdio> #incl