#include <iostream>
#include <cstring>
#include <cstdio>
#include <cstdlib>
#include <vector>
#include <queue>
#include <stack>
#include <map>
#include <set>
#include <cmath>
#define LL long long
using namespace std;
#include<stdio.h>
#include<iostream>
#include<math.h>
#include<string.h>
#include<algorithm>
using namespace std;
const int MAXN = 10010;
int rmq[2*MAXN];//建立RMQ的数组
//***************************
//ST算法,里面含有初始化init(n)和query(s,t)函数
//点的编号从1开始,1-n.返回最小值的下标
//***************************
struct ST
{
int mm[2*MAXN];//mm[i]表示i的最高位,mm[1]=0,mm[2]=1,mm[3]=1,mm[4]=2
int dp[MAXN*2][20];
void init(int n)
{
mm[0]=-1;
for(int i=1;i<=n;i++)
{
mm[i]=((i&(i-1))==0?mm[i-1]+1:mm[i-1]);
dp[i][0]=i;
}
for(int j=1;j<=mm[n];j++)
for(int i=1;i+(1<<j)-1<=n;i++)
dp[i][j]=rmq[dp[i][j-1]]<rmq[dp[i+(1<<(j-1))][j-1]]?dp[i][j-1]:dp[i+(1<<(j-1))][j-1];
}
int query(int a,int b)//查询a到b间最小值的下标
{
if(a>b)swap(a,b);
int k=mm[b-a+1];
return rmq[dp[a][k]]<rmq[dp[b-(1<<k)+1][k]]?dp[a][k]:dp[b-(1<<k)+1][k];
}
};
//边的结构体定义
struct Node
{
int to,next;
};
/* ******************************************
LCA转化为RMQ的问题
MAXN为最大结点数。ST的数组 和 F,edge要设置为2*MAXN
F是欧拉序列,rmq是深度序列,P是某点在F中第一次出现的下标
*********************************************/
struct LCA2RMQ
{
int n;//结点个数
Node edge[2*MAXN];//树的边,因为是建无向边,所以是两倍
int tol;//边的计数
int head[MAXN];//头结点
bool vis[MAXN];//访问标记
int F[2*MAXN];//F是欧拉序列,就是DFS遍历的顺序
int P[MAXN];//某点在F中第一次出现的位置
int cnt;
ST st;
void init(int n)//n为所以点的总个数,可以从0开始,也可以从1开始
{
this->n=n;
tol=0;
memset(head,-1,sizeof(head));
}
void addedge(int a,int b)//加边
{
edge[tol].to=b;
edge[tol].next=head[a];
head[a]=tol++;
edge[tol].to=a;
edge[tol].next=head[b];
head[b]=tol++;
}
int query(int a,int b)//传入两个节点,返回他们的LCA编号
{
return F[st.query(P[a],P[b])];
}
void dfs(int a,int lev)
{
vis[a]=true;
++cnt;//先加,保证F序列和rmq序列从1开始
F[cnt]=a;//欧拉序列,编号从1开始,共2*n-1个元素
rmq[cnt]=lev;//rmq数组是深度序列
P[a]=cnt;
for(int i=head[a];i!=-1;i=edge[i].next)
{
int v=edge[i].to;
if(vis[v])continue;
dfs(v,lev+1);
++cnt;
F[cnt]=a;
rmq[cnt]=lev;
}
}
void solve(int root)
{
memset(vis,false,sizeof(vis));
cnt=0;
dfs(root,0);
st.init(2*n-1);
}
}lca;
bool flag[MAXN];
int main()
{
int T, N, u, v;
scanf("%d", &T);
while(T--)
{
scanf("%d", &N);
lca.init(N);
memset(flag, true, sizeof(flag));
for(int i=1;i<N;i++)
{
scanf("%d%d", &u, &v);
flag[v] = false;
lca.addedge(u, v);
}
int root;
for(int i=1;i<=N;i++) if(flag[i])
{
root = i;
break;
}
lca.solve(root);
scanf("%d%d", &u, &v);
printf("%d\n", lca.query(u,v));
}
return 0;
}
POJ 1330 (LCA的在线算法,转RMQ用ST算法)
最新推荐文章于 2020-02-21 10:57:01 发布