[BZOJ1602&BZOJ1787&BZOJ2144]树上LCA的算法巩固练习

简述求LCA的倍增算法

  对于树上的所有节点,我们可以很轻松地通过dfs求出其直接的父亲节点以及其深度

  通过类似RMQ的原理我们可以处理出每个节点的第2^i个父亲

  //这个过程既可以在dfs之后双重循环建也可以像树剖模板里那样dfs里直接建

  //个人比较推荐后者,会少掉一些不必要的运算,但由于log算法的优越性使得它们实际差别不大

  如图u,v为题目中给的两个节点,我们要做的第一步是将u,v调整到同一深度

  做法很简单,只需要用2^i从大到小逼近答案

  调整到同一深度以后两个节点共同前进,做法和上面调整深度时一样

  细节:

    当u,v的深度刚开始就相同时一定要特判,因为运算到ln(0)会出错



BZOJ1602 简单的LCA模板题

BZOJ1787 题目大意是让我们求出树上三个点到同一点的边权加和最小,输出那个点和最小边权


  我们通过上面的例子可以发现,并不是三点的LCA就是边权和最小的

  因为图上的红边如果走到红色节点的时候会走两次,而走到绿色节点时只走一次,其他路径上的节点各一次

  发现其实和树的重心有点关系...然后就想到了暴力滚粗的ZJOI Day1T1

  其实这道题没那么麻烦...因为只有三个点,很容易想到最终的答案一定是其中两个点的LCA

  那么枚举三次就可以了,先将其中两个点做一次LCA,求出路径边权和,再将这个新求出的点和剩下的点做LCA,求路径和

  第一问的答案就是第一次LCA后求出的那个点

BZOJ2144

  几乎看不出来和LCA有什么关系的题...

  但是想出来了之后觉得这个思路简直太好了...

  对于一个状态我们用三元组表示(x,y,z)

  一种转移是从外面的点跳向中间,而显然由于题目中“只能跳过一颗棋子”的限制所以只有一种方法

  而从中间的点跳向两边就有两种方式

  按照以前的思路,这里就直接BFS敲起来了..

  我们考虑每一个三元组不停地向中间跳一定有一个最终状态

  而这个最终状态向外跳又能产生一系列形如二叉树的状态

  我们将向外跳定义为向儿子状态的连边,向上条定义为向父亲节点的连边

  两个状态的树上路径长度正好的题目中要求的内容

  而第一问只需判断根节点的状态是否一样就可以了

  但对于10^9的数据,想到这里显然还不够

  我们面临着两个问题,一个是depth怎么求(数组根本开不下状态也枚举不完),第二个是第2^i个父亲怎么求

  我们发现这两个问题是有联系的

  考虑一个状态(x,y,z),设t1=y-x,t2=z-y

  当t1>t2时,显然是右边的z往左跳,但是可以跳几步呢?我们解不等式即可得出:(t1-1)div t2步

  当t2>t1时同理

  每次更新t1,t2我们发现它实际上是一个辗转相除的过程,也就是没有几步就可以到达根节点

  也可以根据这个过程叠加出深度

  也可以根据这个过程,算出已知状态的第2^i个父亲状态

  这样一来,这个问题就差不多解决了

  最后一个细节,读进来的状态是无序的,要排序后再做/w\

时间: 2024-08-29 11:12:51

[BZOJ1602&BZOJ1787&BZOJ2144]树上LCA的算法巩固练习的相关文章

hihocoder1069最近公共祖先·三(LCA在线算法--DFS+RMQ-ST)

树上任意两点的最近祖先,必定就是这两个节点的最短路径上深度最小的那个点. 例如:下图中,节点7和5,其最短路径为7--4--1--5, 这条路径上深度最小的点为节点1,其深度为1.节点1即为节点7和5的LCA. 因此,要找到任意两个节点的LCA,只需要先找到上述最短路径,再找到最短路径中深度最小的点.而这下面所述LCA在线算法所做的事. LCA在线算法描述(以上图为例): 1.获得“最短路径”(并不是真正的一条路径,包含其他节点,但不影响算法的正确性) 采用DFS遍历整棵树,得到以下数据: (1

我对最近公共祖先LCA(Tarjan算法)的理解

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

HDU 2874 LCA离线算法

Connections between cities Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 4477    Accepted Submission(s): 1284 Problem Description After World War X, a lot of cities have been seriously damage

最近公共祖先LCA(Tarjan算法)的思考和算法实现——转载自Vendetta Blogs

最近公共祖先LCA(Tarjan算法)的思考和算法实现 LCA 最近公共祖先 Tarjan(离线)算法的基本思路及其算法实现 小广告:METO CODE 安溪一中信息学在线评测系统(OJ) //由于这是第一篇博客..有点瑕疵...比如我把false写成了flase...看的时候注意一下! //还有...这篇字比较多 比较杂....毕竟是第一次嘛 将就将就 后面会重新改!!! 首先是最近公共祖先的概念(什么是最近公共祖先?): 在一棵没有环的树上,每个节点肯定有其父亲节点和祖先节点,而最近公共祖先

HDU 2586 LCA离线算法 tarjan算法

LCA tarjan算法模板题 题意:给一个无根树,有q个询问,每个询问两个点,问两点的距离. 用tarjan离线算法算出每个询问的两点的最近公共祖先 ans[i]=dis[x[i]]+dis[y[i]]-2*dis[z[i]]; //  x[i],y[i]分别存储每次询问的两点,z[i]存储这两点的最近公共祖先 #include "stdio.h" #include "string.h" int tot,n,m; int f[40010],x[40010],y[4

HDU 4547 LCA倍增算法

CD操作 Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others)Total Submission(s): 1111    Accepted Submission(s): 297 Problem Description 在Windows下我们可以通过cmd运行DOS的部分功能,其中CD是一条很有意思的命令,通过CD操作,我们可以改变当前目录. 这里我们简化一下问题,假设只有一个根目录,

LCA倍增算法

LCA 算法是一个技巧性很强的算法. 十分感谢月老提供的模板. 这里我实现LCA是通过倍增,其实就是二进制优化. 任何一个数都可以有2的阶数实现 例如16可以由1 2 4 8组合得到 5可以由1 2 4 组合得到 便于读者理解 我放一道例题吧 Problem F: 挑战迷宫 Description 小翔和小明正在挑战一个神奇的迷宫.迷宫由n个房间组成,每个房间的编号为1~n,其中1号房间是他们俩初始位置, 所有房间一共由n-1条路连接,使得房间两两之间能够相互达到(构成一棵树),每条路的长度为W

最近公共祖先 LCA 倍增算法

倍增算法可以在线求树上两个点的LCA,时间复杂度为nlogn 预处理:通过dfs遍历,记录每个节点到根节点的距离dist[u],深度d[u] init()求出树上每个节点u的2^i祖先p[u][i] 求最近公共祖先,根据两个节点的的深度,如不同,向上调整深度大的节点,使得两个节点在同一层上,如果正好是祖先结束,否则,将连个节点同时上移,查询最近公共祖先. void dfs(int u){ for(int i=head[u];i!=-1;i=edge[i].next){ int to=edge[i

bzoj 1787 && bzoj 1832: [Ahoi2008]Meet 紧急集合(倍增LCA)算法竞赛进阶指南

题目描述 原题连接 Y岛风景美丽宜人,气候温和,物产丰富. Y岛上有N个城市(编号\(1,2,-,N\)),有\(N-1\)条城市间的道路连接着它们. 每一条道路都连接某两个城市. 幸运的是,小可可通过这些道路可以走遍Y岛的所有城市. 神奇的是,乘车经过每条道路所需要的费用都是一样的. 小可可,小卡卡和小YY经常想聚会,每次聚会,他们都会选择一个城市,使得3个人到达这个城市的总费用最小. 由于他们计划中还会有很多次聚会,每次都选择一个地点是很烦人的事情,所以他们决定把这件事情交给你来完成. 他们