【Tarjan】【割边割点】HDU4738 —— Caocao's Bridges

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/qq_34283998/article/details/82986925

题目传送门

讨论版是个好地方,本题三大坑都有说

比较水的Tarjan基础题,大概就是求一个权值最小的割边,水过去.
但是,有重边,要注意处理;有可能不连通,这个时候不需要有人去炸桥;最后答案是0,要输出1,因为需要一个人扛炸药过去.

#include<cstdio>
#include<algorithm>
#include<vector>
using namespace std;

const int MAXN=1005;
const int MAXM=1e6+5;
const int INF=0x3f3f3f3f;

int n,m,ncnt,ans;
int dfn[MAXN],low[MAXN];
int ban[MAXM];
struct node{int v,val,id;};
vector<node> E[MAXN];

void Tarjan(int u,int fa=0){
    dfn[u]=low[u]=++ncnt;
    for(int i=0;i<(int)E[u].size();i++){
        int v=E[u][i].v,val=E[u][i].val,id=E[u][i].id;
        if(ban[id]) continue;
        if(!dfn[v]){
            ban[id]=1;
            Tarjan(v,u);
            low[u]=min(low[u],low[v]);
            if(low[v]>dfn[u])
                ans=min(ans,val);
        }
        low[u]=min(low[u],dfn[v]);
    }
}

int main(){
    while(~scanf("%d%d",&n,&m)&&(n||m)){
        ncnt=0;
        for(int i=1;i<=n;i++) dfn[i]=low[i]=0,E[i].clear();
        for(int i=1;i<=m;i++){
            int u,v,val;
            scanf("%d%d%d",&u,&v,&val);
            ban[i]=0;
            E[u].push_back((node){v,val,i});
            E[v].push_back((node){u,val,i});
        }
        ans=INF;
        Tarjan(1);

        bool flag=0;
        for(int i=1;i<=n;i++) if(!dfn[i]){flag=1;break;}
        if(flag){puts("0");continue;}
        if(!ans) ans=1;
        printf("%d\n",ans==INF?-1:ans);
    }
}

猜你喜欢

转载自blog.csdn.net/qq_34283998/article/details/82986925