LCA(最近公共祖先),指对于一棵树上任意两个节点往上走最早都能到达的节点。

求LCA有两种方法,一种是倍增,另一种则是Tarjan。。。。。。。。

Tarjan巧妙利用并查集的思想;

这里的Tarjan是离线算法

先Tarjan下去;

首先有fa[NUM]=num; 

回溯时将子节点的fa变为num

如果对于num的询问中另一个点已经访问;

那他们的LCA为另一个点的find(fa)

原因:&&一个点与另一个点都位于以他们的LCA为根节点的子树中;

如果没有相关点的信息,只说明在该节点的上方,故回溯时把fa的变为父节点; 

这里的find是并查集中的代表元素。。。。。

再处理各种询问

另外,两点距离为dis[x]+dis[y]-2*dis[LCA(x,y)];

附上原题代码及地址http://codevs.cn/problem/2370/

#include<iostream>
#include<cstdlib>
#include<cstdio>
#define N 50001
using namespace std;
struct data{
int to,nxt,ans;
}ask[75001*2];
struct node{
int nxt,to,w;
}edge[N*2+1];
int tot,tot1,n,m;
bool vis[N];
int dis[N],fa[N];
int head[N],head1[75001*2];
int find(int x){if(fa[x]!=x)fa[x]=find(fa[x]);return fa[x];}
void add1(int x,int y){
ask[++tot1].nxt=head1[x];
ask[tot1].to=y;
head1[x]=tot1;
}
void dfs(int num,int hehe){
dis[num]=hehe;
for(int i=head[num];i;i=edge[i].nxt)
if(!dis[edge[i].to]&&edge[i].to){
int to=edge[i].to;
dfs(to,hehe+edge[i].w);
}
}
void Tarjan_LCA_haha(int t){
vis[t]=true;
fa[t]=t;
for(int i=head[t];i;i=edge[i].nxt)
if(!fa[edge[i].to]&&edge[i].to){
int to=edge[i].to;
Tarjan_LCA_haha(to);
fa[to]=t;
}
for(int i=head1[t];i;i=ask[i].nxt)
if(vis[ask[i].to])
ask[i].ans=dis[ask[i].to]+dis[t]-2*dis[find(ask[i].to)];
}
int main(){
scanf("%d",&n);
for(int i=1;i<n;++i){
int a,b,c;
scanf("%d%d%d",&a,&b,&c);
edge[++tot].nxt=head[a];
head[a]=tot;
edge[tot].w=c;
edge[tot].to=b;
edge[++tot].nxt=head[b];
head[b]=tot;
edge[tot].w=c;
edge[tot].to=a;
}
scanf("%d",&m);
for(int i=1;i<=m;++i){
int a,b;
scanf("%d%d",&a,&b);
add1(a,b);
add1(b,a);
}
dfs(0,0);
Tarjan_LCA_haha(0);
for(int i=1;i<=2*m;i+=2)
if(ask[i].ans)printf("%d\n",ask[i].ans);
else printf("%d\n",ask[i+1].ans);
return 0;
}

关于Tarjan(3)——离线LCA的更多相关文章

  1. 【图论】tarjan的离线LCA算法

    百度百科 Definition&Solution 对于求树上\(u\)和\(v\)两点的LCA,使用在线倍增可以做到\(O(nlogn)\)的复杂度.在NOIP这种毒瘤卡常比赛中,为了代码的效 ...

  2. hihoCoder #1067 : 最近公共祖先·二 [ 离线LCA tarjan ]

    传送门: #1067 : 最近公共祖先·二 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 上上回说到,小Hi和小Ho用非常拙劣——或者说粗糙的手段山寨出了一个神奇的网站 ...

  3. poj1470 Closest Common Ancestors [ 离线LCA tarjan ]

    传送门 Closest Common Ancestors Time Limit: 2000MS   Memory Limit: 10000K Total Submissions: 14915   Ac ...

  4. Tarjan算法离线 求 LCA(最近公共祖先)

    本文是网络资料整理或部分转载或部分原创,参考文章如下: https://www.cnblogs.com/JVxie/p/4854719.html http://blog.csdn.net/ywcpig ...

  5. 【BFS】【并查集】【Tarjan】【LCA】Gym - 101173H - Hangar Hurdles

    给你一张地图,给你q次询问,每次问你从A点到B点,最大能移动多大的箱子. 把每个点所能容纳的最大箱子求出来(BFS,八连通,一开始将所有边界点和障碍点入队).然后从大到小排序.然后用并查集将相邻(四联 ...

  6. tarjan算法求LCA

    tarjan算法求LCA LCA(Least Common Ancestors)的意思是最近公共祖先,即在一棵树中,找出两节点最近的公共祖先. 这里我们使用tarjan算法离线算法解决这个问题. 离线 ...

  7. HDU 2586 How far away ? 离线lca模板题

    How far away ? Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)To ...

  8. HDU 5044 离线LCA算法

    昨天写了HDU 3966 ,本来这道题是很好解得,结果我想用离线LCA 耍一把,结果发现离线LCA 没理解透,错了好多遍,终得AC ,这题比起 HDU 3966要简单,因为他不用动态查询.但是我还是错 ...

  9. SPOJ 10628 Count on a tree(Tarjan离线LCA+主席树求树上第K小)

    COT - Count on a tree #tree You are given a tree with N nodes.The tree nodes are numbered from 1 to  ...

随机推荐

  1. Python学习--13 文件I/O

    Python内置了读写文件的函数,用法和C是兼容的. 读写文件前,我们先必须了解一下,在磁盘上读写文件的功能都是由操作系统提供的,现代操作系统不允许普通的程序直接操作磁盘,所以,读写文件就是请求操作系 ...

  2. MyEclipse-Initializing Java Tooling问题

    问题描述: 今天早上打开Eclipse,打开Package Explorer 中项目时,总出现卡死现象. MyEclipse状态栏显示两个任务,(1) Checking for Updates (2) ...

  3. jQuery 学习总结(上)

    第二章:基础选择器 第三章:过滤性选择器 第四章:表单选择器 第五章:jQuery  操作DOM 第六章:jQuery 事件与应用 第七章:jQuery 实现ajax应用

  4. php面向对象(OOP)---- 验证码类

    PHP常用自封装类--验证码类 验证码是众多网站登陆.注册等相关功能不可以或缺的功能,实现展示验证码的方式有很多,这篇文章作者以工作中比较常用的方法进行了封装. 逻辑准备 要实现一个完整的验证码,需要 ...

  5. 前端基本知识(一):W3C标准&&冒泡事件,捕获事件,W3C DOM对象模型,对比分析

    W3C标准是万维网联盟, 其他的可以参考万维网版本的更新内容 一.W3C标准 二.W3C DOM事件 三.冒泡事件 四.捕获事件 一.W3C标准 其实网页是由三分部组成:1.结构(structure) ...

  6. 在Ubuntu12.0至14.04版本之间用Apache搭建网站运行环境

    为了顺利安装各种软件,先更新下系统. apt-get update 安装Apache服务 apt-get install apache2 -y 安装php apt-get install php5 - ...

  7. C++ cout 输出小数点后指定位数

    在C中我们可以使用 printf("%.2lf",a);但在C++中是没有格式操作符的,该如何操作: C++使用setprecision()函数,同时必须包含头文件iomanip, ...

  8. 转: 尽己力,无愧于心 FastReport.Net 常用功能总汇

    FastReport.Net 常用功能总汇   一.常用控件 文本框:输入文字或表达式 表格:设置表格的行列数,输入数字或表达式 子报表:放置子报表后,系统会自动增加一个页面,你可以在此页面上设计需要 ...

  9. winsshfs的快速入手

    之前在公司使用mac ,并且通过mac下的osfuse和sshfs连接,直接将虚拟机的文件目录同步到了本地,并且可以进行实时操作修改,对于写项目,确实是省了很大一部分上传的精力. 于是在自己的win下 ...

  10. JAVA 命令行参数解析,org.apache.commons.cli的使用

    maven依赖引入 <dependency> <groupId>commons-cli</groupId> <artifactId>commons-cl ...