LCA(最近公共祖先)模板
Tarjan版本
/* gyt
Live up to every day */
#pragma comment(linker,"/STACK:1024000000,1024000000")
#include<cstdio>
#include<cmath>
#include<iostream>
#include<algorithm>
#include<vector>
#include<stack>
#include<cstring>
#include<queue>
#include<set>
#include<string>
#include<map>
#include <time.h>
#define PI acos(-1)
using namespace std;
typedef long long ll;
typedef double db;
const int maxn = 1e5+;
const ll maxm = 1e7;
const ll base = ;
const int INF = <<;
const db eps = 1e-;
const ll mod = 1e9+;
struct Edge{
int u, v, next, val;
}edge[maxn*], edge1[maxn*];
int n, m;
int head[maxn], head1[maxn];
int father[maxn], dis[maxn], tol, toll, LCA[maxn];
bool visit[maxn];
int cnt;
//LCA[]最近公共祖先 void init() {
cnt=;
memset(head, -, sizeof(head));
memset(head1, -, sizeof(head1));
tol=toll=;
memset(visit, , sizeof(visit));
memset(LCA, , sizeof(LCA));
memset(dis, , sizeof(dis));
}
void add(int u, int v, int w) {
edge[tol].u=u, edge[tol].v=v,edge[tol].val=w;
edge[tol].next=head[u];
head[u]=tol++;
}
void add1(int u, int v, int w) {
edge1[toll].u=u, edge1[toll].v=v,edge1[toll].val=w;
edge1[toll].next=head1[u];
head1[u]=toll++;
}
int Find(int x) {
if (x!=father[x]) father[x]=Find(father[x]);
return father[x];
}
void tarjan(int u) {
visit[u]=;
father[u]=u;
for (int j=head1[u]; ~j; j=edge1[j].next) {
int v=edge1[j].v;
if (visit[v]) LCA[edge1[j].val]=Find(v);
}
for (int j=head[u]; ~j; j=edge[j].next) {
int v=edge[j].v;
if (!visit[v]) {
dis[v]=dis[u]+edge[j].val;
tarjan(v);
father[v]=u;
//cout<<dis[v]<<endl;
}
}
}
void solve() {
init();
scanf("%d%d", &n, &m);
for (int i=; i<n; i++) {
int a, b, c; scanf("%d%d%d", &a, &b, &c);
add(a, b, c); add(b, a, c);
}
for (int i=; i<m; i++) {
int a, b; scanf("%d%d", &a, &b);
add1(a, b, i); add1(b, a, i);
}
dis[]=;
tarjan();
for(int i=;i<toll;i+=){
printf("%d\n",LCA[edge1[i].val]);
}
}
int main() {
int t = ;
//freopen("in.txt","r",stdin);
// freopen("gcd.out","w",stdout);
scanf("%d", &t);
while(t--)
solve();
return ;
}
倍增版本
/* gyt
Live up to every day */
#pragma comment(linker,"/STACK:1024000000,1024000000")
#include<cstdio>
#include<cmath>
#include<iostream>
#include<algorithm>
#include<vector>
#include<stack>
#include<cstring>
#include<queue>
#include<set>
#include<string>
#include<map>
#include <time.h>
#define PI acos(-1)
using namespace std;
typedef long long ll;
typedef double db;
const int maxn = 1e5+;
const ll maxm = 1e7;
const ll base = ;
const int INF = <<;
const db eps = 1e-;
const ll mod = 1e9+;
struct Edge{
int u, v, next, val;
}edge[maxn*];
int head[maxn], dis[maxn];
int f[maxn], deep[maxn], p[maxn][];
int n, m, cnt; void init() {
int i,j;
for(j=;(<<j)<=n;j++)
for(i=;i<=n;i++)
p[i][j]=-;
for(i=;i<=n;i++)p[i][]=f[i];
for(j=;(<<j)<=n;j++)
for(i=;i<=n;i++)
if(p[i][j-]!=-)
p[i][j]=p[p[i][j-]][j-];//i的第2^j祖先就是i的第2^(j-1)祖先的第2^(j-1)祖先
}
void add(int u, int v, int w) {
edge[cnt].u=u, edge[cnt].v=v;
edge[cnt].next=head[u], edge[cnt].val=w;
head[u]=cnt++;
}
void dfs(int u, int pre, int t) {
deep[u]=t;
f[u]=pre;
for (int i=head[u]; ~i; i=edge[i].next) {
int v=edge[i].v;
if (v!=pre) {
dis[v]=dis[u]+edge[i].val;
dfs(v, u, t+);
}
}
}
int lca(int a, int b) {
int i, j;
if (deep[a]<deep[b]) swap(a, b);
for ( i=; (<<i)<=deep[a]; i++);
i--;
for ( j=i; j>=; j--) {
if (deep[a]-(<<j)>=deep[b]) a=p[a][j];
}
if (a==b) return a;
for ( j=i; j>=; j--) {
if (p[a][j]!=- && p[a][j]!=p[b][j]) {
a=p[a][j], b=p[b][j];
}
}
return f[a];
}
void solve() {
scanf("%d%d", &n, &m);
cnt=;
memset(head, -, sizeof(head));
for (int i=; i<n-; i++) {
int a, b, c; scanf("%d%d%d", &a, &b, &c);
add(a, b, c); add(b, a, c);
}
dis[]=;
dfs(, -, );
init();
for (int i=; i<m; i++) {
int a, b; scanf("%d%d", &a, &b);
int ans=dis[a]+dis[b]-*dis[lca(a, b)];
printf("%d\n", ans);
}
}
int main() {
int t = ;
// freopen("in.txt","r",stdin);
// freopen("gcd.out","w",stdout);
scanf("%d", &t);
while(t--)
solve();
return ;
}
LCA(最近公共祖先)模板的更多相关文章
- LCA最近公共祖先模板(求树上任意两个节点的最短距离 || 求两个点的路进(有且只有唯一的一条))
		
原理可以参考大神 LCA_Tarjan (离线) TarjanTarjan 算法求 LCA 的时间复杂度为 O(n+q) ,是一种离线算法,要用到并查集.(注:这里的复杂度其实应该不是 O(n+q) ...
 - LCA最近公共祖先模板代码
		
vector模拟邻接表: #include<iostream> #include<cstdio> #include<cstring> #include<cma ...
 - lca最短公共祖先模板(hdu2586)
		
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2586 #include<iostream> #include<cstdio> ...
 - LCA(最近公共祖先)之倍增算法
		
概述 对于有根树T的两个结点u.v,最近公共祖先LCA(T,u,v)表示一个结点x,满足x是u.v的祖先且x的深度尽可能大. 如图,3和5的最近公共祖先是1,5和2的最近公共祖先是4 在本篇中我们先介 ...
 - lca 最近公共祖先
		
http://poj.org/problem?id=1330 #include<cstdio> #include<cstring> #include<algorithm& ...
 - Tarjan算法应用 (割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)问题)(转载)
		
Tarjan算法应用 (割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)问题)(转载) 转载自:http://hi.baidu.com/lydrainbowcat/blog/item/2 ...
 - CodeVs.1036 商务旅行 ( LCA 最近公共祖先 )
		
CodeVs.1036 商务旅行 ( LCA 最近公共祖先 ) 题意分析 某首都城市的商人要经常到各城镇去做生意,他们按自己的路线去做,目的是为了更好的节约时间. 假设有N个城镇,首都编号为1,商人从 ...
 - LCA近期公共祖先
		
LCA近期公共祖先 该分析转之:http://kmplayer.iteye.com/blog/604518 1,并查集+dfs 对整个树进行深度优先遍历.并在遍历的过程中不断地把一些眼下可能查询到的而 ...
 - LCA 近期公共祖先 小结
		
LCA 近期公共祖先 小结 以poj 1330为例.对LCA的3种经常使用的算法进行介绍,分别为 1. 离线tarjan 2. 基于倍增法的LCA 3. 基于RMQ的LCA 1. 离线tarjan / ...
 
随机推荐
- 正则表达式pattern属性
			
你发现自己多久匆匆编写一些正则表达式验证一个特定的文本. 多亏了新的pattern属性,我们可以在标签处直接插入一个正则表达式. <form action="" method ...
 - vue通过ID(参数)修改URL复用同一个页面(组件)不重新加载的问题
			
项目中经常会用到同一个页面,结构是相同的,我只是在vue-router中通过添加参数的方式来区分状态,参数可以在页面跳转时带上params,或者query,但是有一个问题,即使我们修改了参数,URL也 ...
 - 快速了解和使用Photon Server
			
https://blog.csdn.net/qq_36565626/article/details/78710787
 - C++ map与unordered_map
			
map与unordered_map对比 map unordered_map 红黑树(非严格二叉平衡搜索树)实现 哈希表实现 有序 无序 -- 查找时间复杂度为O(1),非常快 空间消耗较大 空间消耗较 ...
 - Java 的静态工厂方法
			
本文转载自:https://www.jianshu.com/p/ceb5ec8f1174 序:什么是静态工厂方法 Effective Java 2.1 静态工厂方法与构造器不同的第一优势在于,它们有名 ...
 - stark组件之搜索【模仿Django的admin】
			
一.先看下django的admin是如何做搜索功能的 配置一个search_fields的列表就可以实现搜索的功能 class testbook(admin.ModelAdmin): # 第一步,定义 ...
 - java_3选择与循环
			
1.三种执行顺序(流程控制语句) 在Java中,有三种执行结构,第一种:顺序结构.第二种:循环结构.第三种:选择结构. 2.顺序结构 自上而下,顺序执行. 3.循环结构 (1)while语句 初始化表 ...
 - MyBatis入门程序(1)
			
一.入门程序: 1.mybatis的配置文件SqlMapConfig.xml 配置mybatis的运行环境,数据源.事务等. <?xml version="1.0" enco ...
 - 15-算法训练 P1103
			
http://lx.lanqiao.cn/problem.page?gpid=T372 算法训练 P1103 时间限制:1.0s 内存限制:256.0MB 编程实现两个复数的运算 ...
 - Aspose.Words五 MergeField
			
通过MegerField来循环,将数据保存到dataset的table中,dataset通过关联datarelation字段来指定主从表关系.模板中通过标签TableStart和TableEnd来框定 ...