求图的强连通分量--tarjan算法

一:tarjan算法详解

◦思想:
◦做一遍DFS,用dfn[i]表示编号为i的节点在DFS过程中的访问序号(也可以叫做开始时间)用low[i]表示i节点DFS过程中i的下方节点所能到达的开始时间最早的节点的开始时间。(也就是之后的深搜所能到达的最小开始时间)初始时dfn[i]=low[i]
◦在DFS过程中会形成一搜索树。在搜索树上越先遍历到的节点,显然dfn的值就越小。
◦DFS过程中,碰到哪个节点,就将哪个节点入栈。栈中节点只有在其所属的强连通分量已经全部求出时,才会出栈。
◦(对于一个强连通分量和一个指出去的有向边,在输出这个强连通分量的时候,那条指出去的边的终点已经作为一个强连通分量统计,并且出栈了。)
◦如果发现某节点u有边连到搜索树(栈)中栈里的节点v,则更新u的low 值为dfn[v](更新为low[v]也可以,更新为low可以算是压缩路径,速度略快)。
◦如果一个节点u已经DFS访问结束,而且此时其low值等于dfn值,则说明u可达的所有节点,都不能到达任何在u之前被DFS访问的节点 ---- 那么该节点u就是一个强连通分量在DFS搜索树中的根。
◦此时将栈中所有节点弹出,包括u,就找到了一个强连通分量
模板:void Tarjan(u) {/*注意小写的tarjan是关键字,还有index也是*/    dfn[u]=low[u]=++index    stack.push(u)
for each (u, v) in E {
        if (v is not visted) {
            tarjan(v) 
            low[u] = min(low[u], low[v]) 
        }
        else if (v in stack) {
            low[u] = min(low[u], dfn[v]) /*这里是low[u] = min(low[u], low[v])也是可以的,因为如果这个点被访问了,而且仍然待在栈中,说明u---v是一条回边,那么这里用dfn[v]
low[v]都是相同了的*/
}
} if (dfn[u] == low[u])
{ //u是一个强连通分量的根 repeat v = stack.pop
print v
until (u== v) } //退栈,把整个强连通分量都弹出来 } //复杂度是O(E+V)的

 

例题:1001. [WZOI2011 S3] 消息传递(来源sojs.tk)

★★   输入文件:messagew.in   输出文件:messagew.out   简单对比
时间限制:1 s   内存限制:128 MB

Problem 2 消息传递 (messagew.pas/c/cpp)
问题描述
WZland开办了一个俱乐部(这里面可以干任何的事情),这引来了许多的人来加入。俱乐部的人数越来越多,关系也越来越复杂……
俱乐部的人来自各个地方,为了增加友谊,俱乐部举行了一次晚会。晚会上又进行了一个传话游戏,如果A认识B,那么A收到某个消息,就会把这个消息传给B,以及所有A认识的人(如果A认识B,B不一定认识A),所有人从1到N编号。
现在给出所有“认识”关系,俱乐部的负责人WZland的国王想知道一个十分简单的问题:如果A发布一条新消息,那么会不会经过若干次传话后,这个消息传回给了A,1≤A≤N。但是WZland的国王是出了名的数学差,幸好的是你在他的身边,于是他就将这个问题交给你来解决。
输入格式
输入数据中的第一行是两个数N和M,两数之间有一个空格,表示人数和认识关系数。
接下来的M行,每行两个数A和B,表示A认识B(1A, BN,AB)。
输出格式
输出文件中一共有N行,每行一个字符“T”或“F”。第i行如果是“T”,表示i发出一条新消息会传回给i;如果是“F”,表示i发出一条新消息不会传回给i。
样例输入输出
message.in 
4 6
1 2
2 3
4 1
3 1
1 3
2 3
message.out
T
T
T
F
数据规模
对于30%的数据,N≤1000,M≤20000;
对于50%的数据,N≤10000,M≤100000;
对于100%的数据,N≤100000,M≤200000;
认识关系可能会重复给出。
时间限制
1s
解析:”认识关系可能会重复给出。“根本不用处理,不会影响答案的。
代码及其分析:
#include<iostream>
using namespace std;
#include<cstdio>
#define N 100100
#include<vector>
vector<int>G[N];
vector<int>ans[N];
int clac=0;
bool inzhan[N];
#include<stack>
#include<cstring>
stack<int>s;
int low[N],dfn[N];
bool flag[N]={0},ok[N]={0};
int n,m;
int Index;
void input()
{
    Index=0;
    scanf("%d%d",&n,&m);
    int a,b;
    for(int i=1;i<=m;++i)
    {
        scanf("%d%d",&a,&b);
        G[a].push_back(b);
    }
    memset(inzhan,false,sizeof(inzhan));
    memset(dfn,-1,sizeof(dfn));
    memset(low,-1,sizeof(low));
}
void Tarjan(int k)
{
    int  j;
    dfn[k]=low[k]=++Index;/*tarjan过程,记录到达该点的时间,和从该点出发能到达的时间最小的点*/
    inzhan[k]=true;/*入栈*/
    s.push(k);
    for(int i=0;i<G[k].size();++i)
    {
        int tp=G[k][i];/*邻接表储存,G[k][i]表示从k点出发的第i条边的终点编号*/
        if(low[tp]==-1)
        {
            Tarjan(tp);
            low[k]=min(low[k],low[tp]);
        }
        else if(inzhan[tp])
        low[k]=min(low[tp],low[k]);
        /*整理总共有三种情况:未被访问,访问了但是是不是强连通分量还不知道(也就是仍在栈中),访问了已经出栈了(这种不用处理,因为如果这个点在好几个强连通分量中,那么他此时一定没有出栈。因为深搜嘛)*/
    }
    if(low[k]==dfn[k])/*low[k]==dfn[k]是这个强连通分量的标志,也就是起始位置,不能再向上找了*/
    {
        int l;
        clac++;/*强连通分量个数*/
        do{
        l=s.top();
        s.pop();
        ans[clac].push_back(l);
        inzhan[l]=false;
        }while(l!=k);
       if(ans[clac].size()>1)
       flag[clac]=true;/*记录这个强连通分量中的所有点是可以传话成功的,下面在重标记每一个点*/
    }
}
void OUT()
{
    for(int i=1;i<=clac;++i)
    if(flag[i])
    {
        for(int j=0;j<ans[i].size();++j)
        ok[ans[i][j]]=true;
    }
    for(int i=1;i<=n;++i)
    if(ok[i])
    printf("T\n");
    else printf("F\n");
}
int main()
{
    freopen("messagew.in","r",stdin);
    freopen("messagew.out","w",stdout);
    input();
    for(int i=1;i<=n;++i)
    if(dfn[i]==-1)
    Tarjan(i);
    OUT();
    fclose(stdin);
    fclose(stdout);
    return 0;
}
View Code

 

posted @ 2016-04-17 23:17  csgc0131123  阅读(1746)  评论(0编辑  收藏  举报