LCA(Least Common Ancestors)

简介: LCA(Least Common Ancestors)

倍增算法(在线)

#include<iostream>
#include<math.h>
#include<string.h>
#define MAXN 10005
using namespace std;
int head[MAXN*2],cnt;
struct edge {
  int to;
  int next;
} e[MAXN*2];//链式向前星的边
int N,d[MAXN],f[MAXN][30],n;
//N是最远跳2的多少次,d[]是每个点的深度,f[i][j]是i点的2^j祖先,n是有多少点
void init() {
  N=(int)(log(n)/log(2));
  cnt=0;
  for(int i=0; i<MAXN; i++) {
    head[i]=-1;
  }
  memset(f,0,sizeof(f));
}
void add(int u,int v) { //链式向前星
  e[cnt].to=v;
  e[cnt].next=head[u];
  head[u]=cnt++;
}
void dfs(int u,int fa,int dep) {
  d[u]=dep;//深度
  f[u][0]=fa;
  for(int i=1; i<=N; i++) {
    f[u][i]=f[f[u][i-1]][i-1];
  }//更新u点到root路径上的祖先
  for(int i=head[u]; i!=-1; i=e[i].next) {
    int v=e[i].to;
    if(v==fa)continue;//以防万一
    dfs(v,u,dep+1);
  }
}//用来求每个点的深度,和f[][]
int lca(int a,int b) {
  if(d[a]>d[b]) {
    swap(a,b);//b为深
  }
  for(int i=N; i>=0; i--) {
    if(d[f[b][i]]>=d[a]) {
      b=f[b][i];
    }
  }//b和a一样深
  if(a==b)return a;//a,b同链
  for(int i=N; i>=0; i--) {
    if( f[a][i] != f[b][i]) { //不相同就跳
      a=f[a][i];
      b=f[b][i];
    }
  }
  return f[a][0];//lca就是父节点
}
int main() {
  int t;
  cin>>t;
  while(t--) {
    cin>>n;
    init();
    for(int i=1; i<n; i++) {
      int a,b;
      cin>>a>>b;
      add(a,b);
      f[b][0]=a;//b的父是a
    }
    for(int i=1; i<n; i++) {
      if(f[i][0]==0) {
        dfs(i,0,1);//找到根结点
        break;
      }
    }
    int a,b;
    cin>>a>>b;
    cout<<lca(a,b)<<endl;
  }
}

Tarjan算法 (离线)

#include<iostream>
#include<math.h>
#include<string.h>
#define MAXN 10005
using namespace std;
int head[MAXN*2],cnt;
bool vis[MAXN];//判断是否访问过了 
bool is_root[MAXN];//从根结点开始dfs 
int f[MAXN];//并查集f[] 
int sox,soy,ans;
struct edge {
  int to;
  int next;
} e[MAXN*2];
void add(int u,int v) { //链式向前星
  e[cnt].to=v;
  e[cnt].next=head[u];
  head[u]=cnt++;
}
void init() {
  cnt=0;
  for(int i=0; i<MAXN; i++) {
    head[i]=-1;
    f[i]=i;
  }
  memset(vis,false,sizeof(vis));
  memset(is_root,true,sizeof(is_root));
}
int find(int x) { //查询祖先节点
  if(x!=f[x])
    f[x]=find(f[x]);
  return f[x];
}
void join(int x,int y) { //合并小子树 
  int fx=find(x),fy=find(y);
  if(fx!=fy)
    f[fy]=fx;
}
void lca(int u){
  for(int i=head[u]; i!=-1; i=e[i].next) {
    int v=e[i].to;
    lca(v);
    join(u,v);
    vis[v]=true;
  }
  if(sox==u &&vis[soy] ){ //子树的父节点就是LCA 
    ans=find(soy);
  }
  if(soy==u &&vis[sox] ){
    ans=find(sox);
  }
}
int main() {
  int t;
  cin>>t;
  while(t--) {
    init();
    int n;
    cin>>n;
    for(int i=1; i<n; i++) {
      int a,b;
      cin>>a>>b;
      add(a,b);
      is_root[b]=false;
    }
    int root;
    for(int i=1; i<=n; i++){
      if(is_root[i]==true)
        root=i;
    }
    cin>>sox>>soy;
    lca(root);
    cout<<ans<<endl;
  }
}
目录
相关文章
Leetcode 236. Lowest Common Ancestor of a Binary Tree
根据LCA的定义,二叉树中最小公共祖先就是两个节点p和q最近的共同祖先节点,LCA的定义没什么好解释的,主要是这道题的解法。
47 0
|
8月前
poj-1458-Common Subsequence
poj-1458-Common Subsequence
39 0
|
Android开发 Kotlin
No signature of method: build_dr75kj88i2pi195a6zalvt5yu.android() is applicable for argument types
No signature of method: build_dr75kj88i2pi195a6zalvt5yu.android() is applicable for argument types
344 0
POJ-1458,Common Subsequence(经典DP)
POJ-1458,Common Subsequence(经典DP)
【1143】Lowest Common Ancestor (30分)
【1143】Lowest Common Ancestor (30分) 【1143】Lowest Common Ancestor (30分)
97 0
|
算法
【POJ 1330 Nearest Common Ancestors】LCA问题 Tarjan算法
题目链接:http://poj.org/problem?id=1330 题意:给定一个n个节点的有根树,以及树中的两个节点u,v,求u,v的最近公共祖先。 数据范围:n [2, 10000] 思路:从树根出发进行后序深度优先遍历,设置vis数组实时记录是否已被访问。
1261 0
|
算法 存储
【HDU 4547 CD操作】LCA问题 Tarjan算法
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4547 题意:模拟DOS下的cd命令,给出n个节点的目录树以及m次查询,每个查询包含一个当前目录cur和一个目标目录tar,返回从cur切换到tar所要使用的cd命令次数: 注意这里的cd命令是简化版,只能进行如下两种操作:   1.
982 0
[LeetCode] Lowest Common Ancestor of a Binary Tree
Well, a follow-up for the problem Lowest Common Ancestor of a Binary Search Tree. However, this time you cannot figure out which subtree the given nodes lie in according to their values.
804 0