一、前人种树

博客:最近公共祖先 LCA 倍增法

博客:浅谈倍增法求LCA

二、沙场练兵

题目:POJ 1330 Nearest Common Ancestors

代码:

  1. const int MAXN = 10010;
  2. const int DEG = 20;
  3.  
  4. struct Edge
  5. {
  6. int to,next;
  7. }edge[MAXN*2];
  8. int head[MAXN],tot;
  9. void addedge(int u,int v)
  10. {
  11. edge[tot].to = v;
  12. edge[tot].next = head[u];
  13. head[u] = tot++;
  14. }
  15. void init()
  16. {
  17. tot = 0;
  18. memset(head,-1,sizeof(head));
  19. }
  20. int fa[MAXN][DEG];//fa[i][j]表示结点i的第2^j个祖先
  21. int deg[MAXN];//深度数组
  22.  
  23. void BFS(int root)
  24. {
  25. queue<int>que;
  26. deg[root] = 0;
  27. fa[root][0] = root;
  28. que.push(root);
  29. while(!que.empty())
  30. {
  31. int tmp = que.front();
  32. que.pop();
  33. for(int i = 1;i < DEG;i++)
  34. fa[tmp][i] = fa[fa[tmp][i-1]][i-1];
  35. for(int i = head[tmp]; i != -1;i = edge[i].next)
  36. {
  37. int v = edge[i].to;
  38. if(v == fa[tmp][0])continue;
  39. deg[v] = deg[tmp] + 1;
  40. fa[v][0] = tmp;
  41. que.push(v);
  42. }
  43.  
  44. }
  45. }
  46. int LCA(int u,int v)
  47. {
  48. if(deg[u] > deg[v])swap(u,v);
  49. int hu = deg[u], hv = deg[v];
  50. int tu = u, tv = v;
  51. for(int det = hv-hu, i = 0; det ;det>>=1, i++)
  52. if(det&1)
  53. tv = fa[tv][i];
  54. if(tu == tv)return tu;
  55. for(int i = DEG-1; i >= 0; i--)
  56. {
  57. if(fa[tu][i] == fa[tv][i])
  58. continue;
  59. tu = fa[tu][i];
  60. tv = fa[tv][i];
  61. }
  62. return fa[tu][0];
  63. }
  64. bool flag[MAXN];
  65. int main()
  66. {
  67. int T;
  68. int n;
  69. int u,v;
  70. scanf("%d",&T);
  71. while(T--)
  72. {
  73. scanf("%d",&n);
  74. init();
  75. memset(flag,false,sizeof(flag));
  76. for(int i = 1;i < n;i++)
  77. {
  78. scanf("%d%d",&u,&v);
  79. addedge(u,v);
  80. addedge(v,u);
  81. flag[v] = true;
  82. }
  83. int root;
  84. for(int i = 1;i <= n;i++)
  85. if(!flag[i])
  86. {
  87. root = i;
  88. break;
  89. }
  90. BFS(root);
  91. scanf("%d%d",&u,&v);
  92. printf("%d\n",LCA(u,v));
  93. }
  94. return 0;
  95. }

LCA(最近公共祖先)——LCA倍增法的更多相关文章

  1. 最近公共祖先 LCA 倍增算法

          树上倍增求LCA LCA指的是最近公共祖先(Least Common Ancestors),如下图所示: 4和5的LCA就是2 那怎么求呢?最粗暴的方法就是先dfs一次,处理出每个点的深度 ...

  2. LCA(最近公共祖先)之倍增算法

    概述 对于有根树T的两个结点u.v,最近公共祖先LCA(T,u,v)表示一个结点x,满足x是u.v的祖先且x的深度尽可能大. 如图,3和5的最近公共祖先是1,5和2的最近公共祖先是4 在本篇中我们先介 ...

  3. 【lhyaaa】最近公共祖先LCA——倍增!!!

    高级的算法——倍增!!! 根据LCA的定义,我们可以知道假如有两个节点x和y,则LCA(x,y)是 x 到根的路 径与 y 到根的路径的交汇点,同时也是 x 和 y 之间所有路径中深度最小的节 点,所 ...

  4. lca最近公共祖先(模板)

    洛谷上的lca模板题--传送门 学了求lca的tarjan算法(离线),在洛谷上做模板题,结果后三个点超时. 又把询问改成链式前向星,才ok. 这个博客,tarjan分析的很详细. 附代码-- #in ...

  5. POJ 1470 Closest Common Ancestors(最近公共祖先 LCA)

    POJ 1470 Closest Common Ancestors(最近公共祖先 LCA) Description Write a program that takes as input a root ...

  6. poj 1330 LCA最近公共祖先

    今天学LCA,先照一个模板学习代码,给一个离线算法,主要方法是并查集加上递归思想. 再搞,第一个离线算法是比较常用了,基本离线都用这种方法了,复杂度O(n+q).通过递归思想和并查集来寻找最近公共祖先 ...

  7. 最近公共祖先(LCA)的三种求解方法

    转载来自:https://blog.andrewei.info/2015/10/08/e6-9c-80-e8-bf-91-e5-85-ac-e5-85-b1-e7-a5-96-e5-85-88lca- ...

  8. LCA 近期公共祖先 小结

    LCA 近期公共祖先 小结 以poj 1330为例.对LCA的3种经常使用的算法进行介绍,分别为 1. 离线tarjan 2. 基于倍增法的LCA 3. 基于RMQ的LCA 1. 离线tarjan / ...

  9. 【图论算法】LCA最近公共祖先问题

    LCA模板题https://www.luogu.com.cn/problem/P3379题意理解 对于有根树T的两个结点u.v,最近公共祖先LCA(u,v)表示一个结点x,满足x是u.v的祖先且x的深 ...

  10. 最近公共祖先LCA(Tarjan算法)的思考和算法实现

    LCA 最近公共祖先 Tarjan(离线)算法的基本思路及其算法实现 小广告:METO CODE 安溪一中信息学在线评测系统(OJ) //由于这是第一篇博客..有点瑕疵...比如我把false写成了f ...

随机推荐

  1. mysql数据去重复distinct、group by

    使用distinct 和group by都可以实现数据去重. select distinct 字段 group by 一般放在where条件后

  2. DBCacheServer服务升级

    前段时间完成了该服务的设计的功能,花了很多时间和经历,最终完成了一个版本,已经测试了:现在后期再次在以前的基础上,完成了一些扩展. 1.扩展了内存存储 最初版本只是采用了gauva cache进行存储 ...

  3. 虚拟环境管理之virtualenvwrapper

    上一篇写了下在linux上使用python的虚拟环境, 干脆把virtualenvwrapper也写一下 1.为什么要用virtualenvwrapper virtualenv 的一个最大的缺点就是: ...

  4. PHP基础 (麦子学院 第二阶段)

    zendstudio 10.0破解版,新建完项目后,首先修改项目的编码方式,统一改成utf-8 (选中项目,再右键properties:Text file encoding).修改字体大小. apac ...

  5. C语言关于指针的注意事项

    一.指针的四个关键概念1.指针的类型2.指针指向的类型3.指针的值,也就是指针指向的地址4.指针自己所占用的内存空间注意:指针变量所存的内容就是内存的地址编号! 例如:int **pp = NULL; ...

  6. python线程与进程小结

    传统方式是调用2个方法执行1个任务,方法按顺序依次执行 # -*- coding:utf-8 -*- import threading import time def run(n): print('t ...

  7. 通过SVI实现VLAN间通信

    两个不同网段的计算机与三层交换机直连,通过SVI实现VLAN间通信vlan 1 //几个不同网段就创建几个VLANvlan 2 int f0/1 //划分VLANswitchport mode acc ...

  8. 多线程编程之Apue3rd_Chapter15.10之posix信号量

    看了APUE的chapter15,只重点看了15.10,学习了posix信号量.Posix信号量比起xsi信号量的优点是性能更好,在Linux3.2.0平台上性能提升很大.其中命名信号量使用方法如下. ...

  9. LINQ巩固

    LINQ巩固 LINQ过滤运算符 Where 基于谓词函数过滤值 测试例子如下: public class TestModel { public string Name { get; set; } p ...

  10. kafka集群部署文档(转载)

    原文链接:http://www.cnblogs.com/luotianshuai/p/5206662.html Kafka初识 1.Kafka使用背景 在我们大量使用分布式数据库.分布式计算集群的时候 ...