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. 《JAVASCRIPT高级程序设计》原生拖放和媒体元素

    一.原生拖放 最早在网页中引入javascript拖放功能的是IE4,当时,网页中只有两种对象可以拖放:图像和某些文本.而现在,几乎网页中的任何元素都可以拖放以及作为放置目标.下面介绍一些与拖放相关的 ...

  2. 启动activity与使用Intent通信机制解析

    我们都知道,一个activity启动另一个activity最简单的方式就是使用startActivity方法: public void startActivity (Intent intent) 但是 ...

  3. javascript 计算两个日期的差值

    代码 Typescript版 /** * TimeSpan just like the class TimpSpan in C# ,represent the time difference * @c ...

  4. phpcms 替换首页

    利用phpcms制作企业站,首先要将静态的企业主页替换成后台可编辑的动态主页. 首先做一个静态的企业站主页: <!DOCTYPE html> <html> <head&g ...

  5. 360随身wifi无法使用临时解决方案大全

       360随身wifi在绝大多数情况下都是可以正常使用的,但在极少数系统或网络环境下可能会出现异常,如系统服务缺失.公司网络限制.少数校园网客户端限制等等:       360攻城师正在积极努力解决 ...

  6. dev简单实现柱状图,曲线图

    1.数据源代码: DataTable dt = new DataTable(); dt.Columns.Add("A"); dt.Columns.Add("B" ...

  7. BZOJ 1004: [HNOI2008]Cards(群论)

    好吧我就是蒟蒻根本没听说过群论(虽说听叉姐说几万年都不会考) 我也讲不太来,直接戳VFK大神的blog啦 = = http://vfleaking.blog.163.com/blog/static/1 ...

  8. ASP.NET MVC 项目直接预览PDF文件

    背景及需求 项目使用的是MVC4框架,其中有一个功能是根据设置生成PDF文件,并在点击时直接预览. 实现过程 1.第一版实现代码: HTML内容 @{ Layout = null; } <!DO ...

  9. ajax跨域问题及解决

    overview ajax是一种创建交互式网页应用的网页开发技术,是一种用于创建快速动态网页的技术,通过在后台与服务器进行少量数据交换.而ajax的跨域问题则是请求了其他项目的接口地址,当协议.子域名 ...

  10. Java和C++的对比

    事实上, Java 本来就是从 C++衍生出来的. C++和 Java 之间仍存在一些显著的差异.可以这样说,这些差异代表着技术的极大进步.一旦我们弄清楚了这些差异,就会理解为什么说 Java 是一种 ...