LCA

最近公共祖先

倍增

在线算法

时间复杂度 \(O((n+q)\log\ n)\)

空间复杂度 \(O(n \log n)\)

fa[x][k] 表示 \(x\)\(2^k\) 祖先

dep[x]表示 \(x\) 的深度

log[x]表示 \(log_2\ x\)

void pre_lca(int now,int father) //预处理LCA
{
    dep[now]=dep[father]+1;
    fa[now][0]=father;
    for(int i=1;(1<<i)<=dep[now];i++)
        fa[now][i]=fa[fa[now][i-1]][i-1];
    for(int i=head[now];i;i=edge[i].next)
        if(edge[i]!=father)
            pre_lca(edge[i].to,now);
}

void pre_log() //预处理log_2 方便查询
{
    log[0]=-1;
    for(int i=1;i<=MAX;i++)
        log[i]=log[i>>1]+1;
}

int lca(int x,int y) //求x y两点的LCA
{
    if(dep[x]<dep[y]) //令x深度比y大
        swap(x,y);
    while(dep[x]>dep[y]) //令x向上跳 直到和y同一高度
        x=fa[x][log[dep[x]-dep[y]]];
    if(x==y) //若x跳到了y上 说明y是x的父结点
        return x;
    for(int i=log[dep[x]];i>=0;i--)
        if(fa[x][i]!=fa[y][i]) //若x y的父结点不同 就可以向上跳
        {
            x=fa[x][i];
            y=fa[y][i];
        }
    return f[x][0]; //此时x y的父结点就是LCA
}

tarjan

01-03 04:21