美文网首页
Tarjan算法求强连通分量

Tarjan算法求强连通分量

作者: Ricardo_Y_Li | 来源:发表于2017-08-29 20:31 被阅读0次
Chtholly

首先先要明确概念:强连通图意为在该图中任意两点间都能够相互到达,而强连通分量即为一个强连通图中的子图,如图中{1,2,3,4}、{5}、{6}即为强连通分量



求强连通分量传统的算法有Kosaraju和Tarjan算法,在这里主要解释Tarjan算法。

算法详解

Tarjan算法是基于对图深度优先搜索的算法,每个强连通分量为搜索树中的一棵子树。搜索时,把当前搜索树中未处理的节点加入一个堆栈,回溯时可以判断栈顶到栈中的节点是否为一个强连通分量。Tarjan算法有点类似于基于后序的深度遍历搜索和并查集的组合,充分利用回溯来解决问题。
在Tarjan当中,主要维护以下几个变量:
dfn[i]维护节点i在该图进行dfs时的次序;
low[i]维护节点i最早能回溯到的节点的编号;
vis[i]维护节点i是否已被访问过;
在Tarjan算法运行时,在我们对图进行dfs的过程中,对于每一个节点u和与它相连的节点v,我们进行如下操作:
如果节点v还未被访问过,那么我们就对v进行dfs,通多定义我们就能够得知,这时low[u]=min(low[u],low[v]);
如果说v已被访问过,即v节点已在栈中,那么这时我们就需要用v的dfn值来更新u的low值,有定义可知low[u]=min(low[u],dfn[v]);
对于一个连通图,我们很容易想到,在该连通图中有且仅有一个节点u的DFN值和low值相等,所以我们在回溯的过程中就能够通过判断节点的low值和DFN值是否相等来判断是否已经找到一个子连通图。由于该连通图中 的DFN值和low值相等的节点是该连通图中第一个被访问到的节点,又根据栈的特性,则该节点在最里面。所以能够通过不停 的弹栈,直到弹出该DFN值和low值相同的节点来弹出该连通图中所有的节点。
另外附上一张其他神犇推演Tarjan的演算图:


关于对在一个强连通图中有且仅有一个点的dfn与low相等这一结论的证明,请访问博客:http://blog.csdn.net/jeryjeryjery/article/details/52829142?locationNum=4&fps=1
下面是C++模板代码实例:
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<vector>
using namespace std;

const int maxn=1e5+10;
vector<int>map[maxn];//用vector来存图
int stack[maxn];//栈 
int dfn[maxn];//节点在dfs时的次序
int low[maxn];//节点最早能够追溯到的节点编号
bool vis[maxn];//节点是否被访问过
int time=0;//时间戳
int top=-1;//栈顶标记
int cnt=0;//强连通分量个数
int n,m;//节点数与边数

inline int read(){//快读
    int x=0,flag=1;
    char c=getchar();
    while(c<'0' || c>'9'){
        if(c=='-')
            flag=-1;
        c=getchar();
    } 
    while(c<='9' && c>='0'){
        x=x*10+(int)(c-'0');
        c=getchar();
    }
    return x*flag;
}

inline void tarjan(int u){
    dfn[u]=low[u]=++time;
    stack[++top]=u;//入栈 
    vis[u]=1;
    int len=map[u].size()-1;//因为在vector中数组从零开始,所以减一
    for(int i=0;i<=len;i++){
        int v=map[u][i];
        if(!vis[v]){//没访问过 
            tarjan(v);
            low[u]=min(low[u],low[v]); 
        }
        else//访问过 
            low[u]=min(low[u],dfn[v]);
    }
    if(dfn[u]==low[u]){//找到强连通分量
        cnt++;
    //开始出栈
        int j;
        do{
            j=stack[top--];
            vis[j]=0; 
        }while(j!=u); 
    } 
}

int main(){
    ios::sync_with_stdio(0);
    memset(dfn,0,sizeof(dfn));//初始化
    memset(low,0,sizeof(low));
    memset(vis,0,sizeof(vis)); 
    n=read();
    m=read();
    for(int i=1;i<=m;i++){
        int x,y;
        x=read();
        y=read();
        map[x].push_back(y);
    }
    tarjan(1);
    cout<<cnt<<endl;
    return 0;
}

相关文章

  • Tarjan算法求强连通分量

    首先先要明确概念:强连通图意为在该图中任意两点间都能够相互到达,而强连通分量即为一个强连通图中的子图,如图中{1,...

  • tarjan

    tarjan:寻找出度为0的强连通分量,并求出该强连通分量中有多少个点。 sig表示的是强连通分量的个数其中col...

  • tarjan

    tarjan寻找出度为0的强连通分量,从小到大输出此强连通分量中的点 poj 2553 The Bottom of...

  • tarjan-寻找图中有多少个强连通分量

    tarjan寻找图中有多少个强连通分量 hdu 1269 迷宫城堡判断图否是属于一个强连通分量

  • 连通分量

    tarjan算法实现,low数组代表该点最先追溯到的编号,dfn数组代表该点按照访问次序编的号。 强连通分量:有向...

  • 图算法(一)遍历,拓扑排序

    本文介绍图的几种基本操作:BFS,DFS,求有向图连通分量的Tarjan算法以及拓扑排序。 图的表示 一张图是由若...

  • Tarjan算法求强联通分量

    Tarjan算法求强联通分量基于对图的DFS: 表示节点在DFS搜索中是第几个被搜索到的(时间戳)。 表示从在DF...

  • Kosaraju算法详解

    Kosaraju算法是干什么的? Kosaraju算法可以计算出一个有向图的强连通分量 什么是强连通分量? 在一个...

  • 强连通分量和Kosaraju算法

    内容概要: 基于深度优先后序遍历的DAG图拓扑排序 强连通分量 求解强连通分量Kosaraju算法 拓扑排序的另一...

  • 图论-----求割点

    求无向连通图割点(java)-------Tarjan算法 在无向连通图中,如果将其中一个点以及所有连接该点的边去...

网友评论

      本文标题:Tarjan算法求强连通分量

      本文链接:https://www.haomeiwen.com/subject/kidadxtx.html