有向图强连通分量Tarjan算法

在https://www.byvoid.com/zhs/blog/scc-tarjan中关于Tarjan算法的描述非常好,转述如下:

首先解释几个概念:

有向图强连通分量:在有向图G中,如果两个顶点间至少存在一条路径,称两个顶点强连通(strongly connected)。

如果有向图G的每两个顶点都强连通,则称G是一个强连通图。

非强连通图有向图的极大强连通子图,成为强连通分量(strongly connected components)。

下图中,子图{1,2,3,4}为一个强连通分量,因为顶点1,2,3,4两两可达,{5},{6}也分别是两个强连通分量。

求强连通分量的Tarjan算法如下

Targin算法

算法的基本思想如下,任选图中的一个点开始进行深度优先搜索,并按访问的顺序对点u进行编号,将结果存在数组index[u]中。另一个数组low,存储u或u的子树能够追溯到最早的栈中节点的次序号。对于一个边(u,v)我们可得

 

low(u) = MIN{
      index(u),
      index(v), v还在栈中,此时还没有算完low(v)
low(v) ,u为v的父节点,v未被访问
}

 

当index[u] = low[u]时,以u为根的子树上的所有节点是一个强连通分量

算法的伪代码如下

tarjan(u)
{
    index[u]=low[u]=++tmp                      // 为节点u设定次序编号和low初值,tmp从0开始
    Stack.push(u)                              // 将节点u压入栈中
    for each (u, v) in E                       // 枚举每一条边
        if (v is not visted)               // 如果节点v未被访问过
            tarjan(v)                  // 继续向下找
            low[u] = min(low[u], low[v])
        else if (v in Stack)                   // 如果节点v还在栈内
            low[u] = min(low[u], index[v])
    if (index[u] == low[u])                      // 如果节点u是强连通分量的根
        repeat
            v = Stack.pop                  // 将v退栈,为该强连通分量中一个顶点
            print v
        until (u == v)
}

 

 

算法应用举例

NODE  INDEX  LOW

1     0     0(未算完)

2    

3    1     1(未算完)

4    

5    2     2(未算完)

6    3              3(未算完)

{6}为一个强连通分量

NODE  INDEX  LOW

1     0     0(未算完)

2    

3    1     1(未算完)

4    

5    2     2

{5}为一个强连通分量

NODE  INDEX  LOW

1     0     0(未算完)

2    

3    1     1(未算完)

4    4              0

 

NODE  INDEX  LOW

1     0     0

2    5     5

3    1     0

4    4              0

{1,2,3,4}为一个强连通分量,算法结束

算法的复杂度为

O(N+M),边数+结点数

posted @ 2017-04-15 20:37  Jason杰  阅读(1010)  评论(0编辑  收藏  举报