LCA(最近公共祖先)

lca即两点到达的第一个公共节点

前两个方法取自:Cold_Chair,树剖方法取自:Frocean


目录

 

暴力往上跳

倍增算法

树剖法


暴力往上跳

先把x,y跳到同一深度,然后同时往上跳,最后fa[x] == fa[y]即找到LCA。

时间复杂度与树的深度有关,则最坏情况就是一条链了。

int dep[maxn];
void dfs(int u){
	for(int i=head[u]; ~i; i=edge[i].next){
		int v = edge[i].to;
		dep[v] = dep[u] + 1;
		fa[v] = u;
		dfs(v);
	}
}
int lca(int x, int y){
	if(dep[x] > dep[y]) swap(x, y);	//使y为深度较大的 
	while(dep[y] > dep[x])	y = fa[y];
	while(x != y)	x = fa[x], y = fa[y];
	return x;
}

倍增算法

类同暴力,也是往上跳着找,只是倍增每次跳2的幂次,更快

预处理幂次:

  1. f(i,j)=f(f(i,j-1),j-1)
  2. 边界  f(i, 0)=1
int dep[maxn];
void dfs(int u){
	for(int i=head[u]; ~i; i=edge[i].next){
		int v = edge[i].to;
		dep[v] = dep[u] + 1;
		fa[v] = u;
		dfs(v);
	}
}
int f[maxn][maxn];
void build(int n){
	dep[1] = 1;
	dfs(1);
	for(int i=1; i<=n; ++i)
		f[i][0] = fa[i];
	for(int j=1; (1<<j)<=n; ++j)
		for(int i=1; i<=n; ++i)
			if(f[i][j-1] != -1)
				f[i][j-1] = f[f[i][j-1]][j-1];
}
int lca(int x, int y){
	if(dep[x] > dep[y])	swap(x, y);
	for(int i=20; i>=0; --i){
		if(dep[f[y][i]] >= dep[x])
			y = f[y][i];
	}
	for(int i=20; i>=0; --i){
		if(f[x][i] != f[y][i])
			x = f[x][i], y = f[y][i];
	}
	return f[x][0];
}

树剖法

树剖入门

通过重链往上跳,最后归于一条链,深度较小时即为结果。

int siz[maxn], dep[maxn], fa[maxn], top[maxn], son[maxn]; //子节点数,深度,父亲,重链头节点,重儿子	
void dfs1(int u){
	siz[u] = 1;
	for(int i=head[u]; ~i; i=edge[i].next){
		int v = edge[i].to;
		dep[v] = dep[u] + 1;
		fa[v] = u;
		dfs(v);
		siz[u] += siz[v];
		if(!son[u] || siz[v] > siz[son[u]])	son[u] = v;
	}
}
void dfs2(int u){
	if(son[u]){
		top[son[u]] = top[u];
		dfs(son[u]);
	}
	for(int i=head[u]; ~i; i=edge[i].next){
		int v = edge[i].to;
		if(v == son[v])	continue;
		top[v] = v;
		dfs(v);
	}
}
int lca(int x, int y){
	while(top[x] != top[y]){
		if(dep[top[x]] > dep[top[y]])	swap(x, y);	//y较深
			y = fa[top[y]];
	}
	if(dep[x] > dep[y])	swap(x, y);	//在一条链上,返回深度较小的一个 
	return x;
}

还有其他求lca方法,等需要时再补充。可查看Cold_Chair

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值