zoukankan      html  css  js  c++  java
  • 无向图的点双连通分量(tarjan模板)

    #include<iostream>
    #include<cstdio>
    #include<algorithm>
    #include<cstring>
    #include<stack>
    #include<vector>
    using namespace std;
    #define maxn 7500
    #define inf 0x3f3f3f3f
    int n,m;
    int g[maxn][maxn];
    int clock;
    int low[maxn],pre[maxn];
    stack<int>s;
    int bc;
    vector<int>bcc[maxn];
    int dfs(int u,int fa){
         low[u]=pre[u]=++clock;
         s.push(u);
         for(int v=1;v<=n;v++){
            if(!g[u][v])continue;
            if(!pre[v]){
                int lowv=dfs(v,u);
                low[u]=min(low[u],lowv);
                if(lowv>=pre[u]){
                    bc++;
                    bcc[bc].clear();
                    int tmp=-1;
                    while(!s.empty()){
                        tmp=s.top();
                        s.pop();
                        bcc[bc].push_back(tmp);
                        if(tmp==u)break;
                    }
                    if(tmp!=-1)s.push(tmp);  //割顶要加回去,随意割顶至少是两个不同双联通分量的公共点
                }
            }
           else if(pre[v]<pre[u]&&fa!=v){
                low[u]=min(low[u],pre[v]);
            }
         }
         return low[u];
    }
    void inital(){
        clock=0;
        bc=0;
        memset(pre,0,sizeof pre);
        memset(low,inf,sizeof low);
        while(!s.empty()){
            s.pop();
        }
    }
    int main()
    {
        int u,v;
        freopen("in.txt","r",stdin);
        while(~scanf("%d%d",&n,&m)){
            inital();
            for(int i=0;i<m;i++){
                scanf("%d%d",&u,&v);
                g[u][v]=g[v][u]=1;
            }
            for(int i=1;i<=n;i++){
                if(!pre[i])dfs(i,-1);
            }
             for(int i=1;i<=n;i++){
                printf("%d %d
    ",pre[i],low[i]);
            }
    
            printf("以下是点联通分量%d:
    ",bc);
            for(int i=1;i<=bc;i++){
                printf("%d:",i);
                for(int j=0;j<bcc[i].size();j++){
                    printf("%d ",bcc[i][j]);
                }
                printf("
    ");
            }
    
        }
    }
    

  • 相关阅读:
    中介者模式
    观察者模式
    javascript深入理解js闭包
    外观模式
    模板方法模式
    浅析C#深拷贝与浅拷贝
    C#浅拷贝与深拷贝区别
    6个重要的.NET概念:栈,堆,值类型,引用类型,装箱,拆箱
    原型模式
    设计模式总结
  • 原文地址:https://www.cnblogs.com/clnchanpin/p/6834751.html
Copyright © 2011-2022 走看看