zoukankan      html  css  js  c++  java
  • LCA倍增算法

    一.倍增算法的前期铺垫

    我们记节点v到根的深度为depth(v)。那么如果节点w是节点u和节点v的最近公共祖先的话,让u往上走(depth(u)-depth(w))步,让v往上走(depth(v)-depth(w))步,都将走到节点w。因此,我们首先让u和v中较深的一个往上走|depth(u)-depth(v)|步,再一起一步步往上走,直到走到同一个节点,就可以在O(depth(u)+depth(v))的时间内求出LCA。

    由于节点的最大深度为n,所以这个方法在最坏的情况下一次查询时间复杂度就要O(n),这显然是不够的。于是我们开始考虑优化。

    二.倍增算法的实现过程

    分析刚才的算法,两个节点到达同一节点后,不论怎么向上走,达到的显然还是同一节点。利用这一点,我们就能够利用二分搜索求出到达最近公共祖先的最小步数了。

    首先我们要进行预处理。对于任意的节点,可以通过fa2[v]=fa[fa[v]]得到其向上走2步到达的顶点,再利用这个信息,又可以通过fa4[v]=fa2[fa2[v]]得到其向上走4步所到的顶点。以此类推,我们可以得到其向上走2^k步所到的顶点fa[v][k],预处理的时间点复杂度为O(nlogn)。

    有了k=floor(logn)以内的所有信息后,就可以进行二分所搜的,每次查询的时间复杂度为O(logn)。

    以poj1330为例

    #include<iostream>
    #include<cstdio>
    #include<vector>
    #include<cmath>
    #include<cstring>
    using namespace std;
    vector<int>v[10010];
    int fa[10010][100];//fa[i][j]表示从i节点走2^j步可以达到的祖先节点 
    int depth[10010];//记录每个节点的深度 
    int n;
    void dfs(int u,int pre,int dep)
    {
        fa[u][0]=pre;
        depth[u]=dep;
        for(int i=0;i<v[u].size();i++)
        {
            int p=v[u][i];
            dfs(p,u,dep+1);
        }
    }
    void init(int root)
    {
        dfs(root,-1,0);
        for(int j=0;(1<<(j+1))<=n;j++)
        {
            for(int i=1;i<=n;i++)
            {
                if(fa[i][j]<0)
                    fa[i][j+1]=-1;
                else
                    fa[i][j+1]=fa[fa[i][j]][j];
            }
        } 
    }
    int LCA(int u,int v)
    {
        if(depth[u]>depth[v])
            swap(u,v);
        int temp=depth[v]-depth[u];
        //cout<<temp<<endl;
        for(int i=0;(1<<i)<=temp;i++)
        {
            if((1<<i)&temp)
            {
                v=fa[v][i];
                //cout<<v<<endl;
            }
            
        }
        if(v==u)
        return u;
        for(int i=int(log(n*1.0));i>=0;i--)
        {
            if(fa[u][i]!=fa[v][i])
            {
                u=fa[u][i];
                v=fa[v][i];
            }
        }
        return fa[u][0];
    }
    int main()
    {
        int T,k,x,y;
        scanf("%d",&T);
        while(T--)
        {
            scanf("%d",&n);
            for(int i=1;i<=n;i++)
                v[i].clear();
            memset(fa,0,sizeof(fa));
            for(int i=1;i<=n-1;i++)
            {
                scanf("%d%d",&x,&y);
                v[x].push_back(y);
                fa[y][0]=x;
            }
            for(int i=1;i<=n;i++)
            {
                if(fa[i][0]==0)
                {
                    k=i;
                    break;
                }
            }
            //cout<<k<<endl;
            init(k);
            scanf("%d%d",&x,&y);
            printf("%d
    ",LCA(x,y));    
        }
    }
  • 相关阅读:
    Markdown快捷笔记
    Linux常用命令
    Git使用
    HTML
    JavaScript-笔记2
    AngularJS-笔记2
    AngularJS-笔记1
    JQuery-笔记
    设置DataGridView的某个单元格为ComboBox
    记录文件浏览历史路径
  • 原文地址:https://www.cnblogs.com/flightless/p/8678689.html
Copyright © 2011-2022 走看看