温州市住房和城乡建设网站,花生壳怎么做网站,自适应网站案例源码,两学一做网站网址大全[有向图强连通分量] 在有向图G中#xff0c;如果两个顶点间至少存在一条路径#xff0c;称两个顶点强连通 (strongly connected)。如果有向图G的每两个顶点都强连通#xff0c;称G是一个强连通图 。非强连通图有向图的极大强连通子图#xff0c;称为强连通分量 (strongly … [有向图强连通分量] 在有向图G中如果两个顶点间至少存在一条路径称两个顶点强连通 (strongly connected)。如果有向图G的每两个顶点都强连通称G是一个强连通图 。非强连通图有向图的极大强连通子图称为强连通分量 (strongly connected components)。 下图中子图{1,2,3,4}为一个强连通分量因为顶点1,2,3,4两两可达。{5},{6}也分别是两个强连通分量。 直接根据定义用双向遍历取交集的方法求强连通分量时间复杂度为O(N^2M)。更好的方法是Kosaraju算法或Tarjan算法两者的时间复杂度都是O(NM)。本文介绍的是Tarjan算法。 [Tarjan算法] Tarjan算法是基于对图深度优先搜索的算法每个强连通分量为搜索树中的一棵子树 。搜索时把当前搜索树中未处理的节点加入一个堆栈回溯时可以判断栈顶到栈中的节点是否为一个强连通分量。 定义DFN(u)为节点u搜索的次序编号(时间戳)Low(u)为u或u的子树能够追溯到的最早的栈中节点的次序号。由定义可以得出 Low(u)Min { DFN(u), Low(v),(u,v)为树枝边u为v的父节点 DFN(v),(u,v)为指向栈中节点的后向边(非横叉边) } 当DFN(u)Low(u)时以u为根的搜索子树上所有节点是一个强连通分量。 算法伪代码如下 tarjan(u) { DFN[u]Low[u]Index // 为节点u设定次序编号和Low初值 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 S) // 如果节点v还在栈内 Low[u] min(Low[u], DFN[v]) if (DFN[u] Low[u]) // 如果节点u是强连通分量的根 repeat v S.pop // 将v退栈为该强连通分量中一个顶点 print v until (u v) } 接下来是对算法流程的演示。 从节点1开始DFS把遍历到的节点加入栈中。搜索到节点u6时DFN[6]LOW[6]找到了一个强连通分量。退栈到uv为止{6}为一个强连通分量。 返回节点5发现DFN[5]LOW[5]退栈后{5}为一个强连通分量。 返回节点3继续搜索到节点4把4加入堆栈。发现节点4向节点1有后向边节点1还在栈中所以LOW[4]1。节点6已经出栈(4,6)是横叉边返回3(3,4)为树枝边所以LOW[3]LOW[4]1。 继续回到节点1最后访问节点2。访问边(2,4)4还在栈中所以LOW[2]DFN[4]5。返回1后发现DFN[1]LOW[1]把栈中节点全部取出组成一个连通分量{1,3,4,2}。 至此算法结束。经过该算法求出了图中全部的三个强连通分量{1,3,4,2},{5},{6}。 可以发现运行Tarjan算法的过程中每个顶点都被访问了一次且只进出了一次堆栈每条边也只被访问了一次所以该算法的时间复杂度为O(NM)。 求有向图的强连通分量还有一个强有力的算法为Kosaraju算法。Kosaraju是基于对有向图及其逆图两次DFS的方法其时间复杂度也是 O(NM)。与Trajan算法相比Kosaraju算法可能会稍微更直观一些。但是Tarjan只用对原图进行一次DFS不用建立逆图更简洁。 在实际的测试中Tarjan算法的运行效率也比Kosaraju算法高30%左右。此外该Tarjan算法与求无向图的双连通分量(割点、桥)的Tarjan算法 也有着很深的联系。学习该Tarjan算法也有助于深入理解求双连通分量的Tarjan算法两者可以类比、组合理解。 求有向图的强连通分量的Tarjan算法是以其发明者Robert Tarjan 命名的。Robert Tarjan还发明了求双连通分量 的Tarjan算法以及求最近公共祖先的离线Tarjan算法在此对Tarjan表示崇高的敬意。 附tarjan算法的C程序 void tarjan(int i) { int j; DFN[i]LOW[i]Dindex; instack[i]true; Stap[Stop]i; for (edge *eV[i];e;ee-next) { je-t; if (!DFN[j]) { tarjan(j); if (LOW[j]LOW[i]) LOW[i]LOW[j]; } else if (instack[j] DFN[j]LOW[i]) LOW[i]DFN[j]; } if (DFN[i]LOW[i]) { Bcnt; do { jStap[Stop--]; instack[j]false; Belong[j]Bcnt; } while (j!i); } } void solve() { int i; StopBcntDindex0; memset(DFN,0,sizeof(DFN)); for (i1;iN;i) if (!DFN[i]) tarjan(i); }