【求无向图的桥,有重边】ZOJ - 2588 Burning Bridges

模板题——求割点与桥

题意,要使一个无向图不连通,输出必定要删掉的边的数量及其编号。求桥的裸题,可拿来练手。

套模板的时候注意本题两节点之间可能有多条边,而模板是不判重边的,所以直接套模板的话,会将重边也当做桥输出,因此要在判断桥的时候加一个判断,即当且仅当两点之间仅有一条边,且满足dfn[cur] < low[i],(cur, i)才是桥。

另外本题节点数为105,用邻接矩阵的话会内存超限,所以我用了了一个multiset存储边及其编号。

代码如下:

 1 #include<cstdio>
 2 #include<cstdlib>
 3 #include<cstring>
 4 #include<vector>
 5 #include<set>
 6 #include<algorithm>
 7 #include<iostream>
 8 using namespace std;
 9 const int MAX_V = 10005;
10 const int MAX_E = 100005;
11 struct Edge
12 {
13     int to, id;
14     Edge() {}
15     Edge(int _t, int _i):to(_t), id(_i) {}
16     bool operator < (const Edge& rhs) const
17     {
18         return to < rhs.to;
19     }
20 };
21 multiset<Edge> edge[MAX_V];
22 multiset<int> tms[MAX_V];
23 vector<int> ans;
24 int vis[MAX_V]; //结点v当前访问状态,0表示未访问,1表示在栈中,2表示已经访问过
25 int dfn[MAX_V]; //结点v被访问时的深度
26 int low[MAX_V]; //结点v可以到达的访问时间最早的祖先的深度
27 void cut_bridge(int cur, int father, int dep, int n) //vertex: 0~n-1
28 {
29     vis[cur] = 1;
30     dfn[cur] = dep;
31     low[cur] = dep;
32     int children = 0;
33     int sz = edge[cur].size();
34     for(multiset<Edge>::iterator it = edge[cur].begin(); it != edge[cur].end(); it++)
35     {
36         int id = (*it).id;
37         int i = (*it).to;
38         if(i != father && vis[i] == 1) //i在当前栈中,说明图中有一个环,用i的深度更新cur的low值;
39         {
40             if(dfn[i] < low[cur]) low[cur] = dfn[i]; //将结点cur可以到达的访问时间最早的祖先的深度更新为结点i被访问时的深度;
41         }
42         if(vis[i] == 0) //i没被访问过,递归访问节点i,并用i的可以到达的最早祖先来更新cur的low值;
43         {
44             cut_bridge(i, cur, dep+1, n);
45             if(low[i] < low[cur]) low[cur] = low[i];
46             if(low[i] > dfn[cur] && tms[cur].count(i) == 1)
47             {
48                 ans.push_back(id);
49             }//判断桥
50         }
51     }
52     vis[cur] = 2;
53 }
54 void init(int n)
55 {
56     ans.clear();
57     memset(vis, 0, sizeof(vis));
58     for(int i = 0; i <= n; i++)
59     {
60         edge[i].clear();
61         tms[i].clear();
62     }
63 }
64 int main()
65 {
66     int T;
67     scanf("%d", &T);
68     for(int kase = 0; kase < T; kase++)
69     {
70         if(kase) printf("\n");
71         int n, m;
72         scanf("%d%d", &n, &m);
73         init(n);
74         for(int i = 0; i < m; i++)
75         {
76             int u, v;
77             scanf("%d%d", &u, &v);
78             Edge e1(v, i+1), e2(u, i+1);
79             edge[u].insert(e1);
80             edge[v].insert(e2);
81             tms[u].insert(v);
82             tms[v].insert(u);
83         }
84         for(int i = 1; i <= n; i++)
85             if(vis[i] == 0)
86                 cut_bridge(i, -1, 1, n);
87
88         sort(ans.begin(), ans.end());
89         printf("%d\n", ans.size());
90         for(int i = 0; i < ans.size(); i++)
91         {
92             if(i) printf(" ");
93             printf("%d", ans[i]);
94         }
95         if(ans.size()) printf("\n");
96     }
97     return 0;
98 }

ZOJ 2588

时间: 2024-08-03 19:29:14

【求无向图的桥,有重边】ZOJ - 2588 Burning Bridges的相关文章

ZOJ 2588 Burning Bridges 求无向图桥 边双连通裸题

题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1588 binshen的板子: #include <stdio.h> #include <string.h> #include <iostream> #include <algorithm> #include <vector> #include <queue> #include <set> #i

ZOJ 2588 Burning Bridges(无向图求割边)

ZOJ 2588 Burning Bridges 链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=2588 题意:给定一个无向图连通图,(其中可能有重边),要求去掉一条边之后,使得整个图不再连通.输出这些符合条件的边的序号. 思路:这就是一个简单的无向图求割边,需要注意的是这个无向图有重边,重边一定不是割边. 代码: /*========================================= 无向图求割点

zoj 2588 Burning Bridges【双连通分量求桥输出桥的编号】

Burning Bridges Time Limit: 5 Seconds      Memory Limit: 32768 KB Ferry Kingdom is a nice little country located on N islands that are connected by M bridges. All bridges are very beautiful and are loved by everyone in the kingdom. Of course, the sys

ZOJ 2588 Burning Bridges(无向连通图求割边)

题目地址:ZOJ 2588 由于数组开小了而TLE了..这题就是一个求无向连通图最小割边.仅仅要推断dfn[u]是否<low[v],由于low指的当前所能回到的祖先的最小标号,增加low[v]大于dfn[u]时,说明v无法通过其它边回到u之前的点.也就是说v假设想要回到u的祖先点.必需要经过u点,那这条边非常明显就是一条割边.这题还要去重边,假如有重边的话.说明怎么销毁哪条边总能通过还有一条边,所以仅仅要有重边.说明这两点之间没有割边. 代码例如以下: #include <iostream&g

ZOJ 2588 Burning Bridges(强连通分量)

题目地址:ZOJ 2588 因为数组开小了而TLE了..这题就是一个求无向连通图最小割边.只要判断dfn[u]是否<low[v],因为low指的当前所能回到的祖先的最小标号,加入low[v]大于dfn[u]时,说明v无法通过其他边回到u之前的点,也就是说v如果想要回到u的祖先点,必须要经过u点,那这条边很明显就是一条割边.这题还要去重边,假如有重边的话,说明怎么销毁哪条边总能通过另一条边,所以只要有重边,说明这两点之间没有割边. 代码如下: #include <iostream> #in

ZOJ 2588 Burning Bridges(判断割边)

题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=2588 题意: Ferry王国是一个漂亮的岛国,一共有N个岛国.M座桥,通过这些桥可以从每个小岛都能 到达任何一个小岛.很不幸的是,最近Ferry王国被Jordan征服了.Jordan决定烧毁所有的桥. 这是个残酷的决定,但是Jordan的谋士建议他不要这样做,因为如果烧毁所有的桥梁,他自己的 军队也不能从一个岛到达另一个岛.因此Jordan决定烧尽可能多的桥,只

ZOJ 2588 Burning Bridges 割边的求解

题目链接: ZOJ2588 题意: 给出一个无向的连通图,问去掉图中的哪些边,都会使图将不连通 题解思路: 割边的求解: 1.需要用到Tarjan算法的框架,首先求出dfn low 两个数组 当递归返回时  判断dfn[u]和low[v]的关系 只有当dfn[u]  < low[v]  的情况下u-v是一条割边(u是关节点 ,且u-v不含重边,即dfn[u] != low[v]) 2.题目中还有出现重边的情况  重边是不可能成为割边的  我们需要对重边进行标记 代码: #include<ios

Hdu 4738【tanjan求无向图的桥】

题目: 曹操在长江上建立了一些点,点之间有一些边连着.如果这些点构成的无向图变成了连通图,那么曹操就无敌了.刘备为了防止曹操变得无敌,就打算去摧毁连接曹操的点的桥.但是诸葛亮把所有炸弹都带走了,只留下一枚给刘备.所以刘备只能炸一条桥. 题目给出n,m.表示有n个点,m条桥. 接下来的m行每行给出a,b,c,表示a点和b点之间有一条桥,而且曹操派了c个人去守卫这条桥. 现在问刘备最少派多少人去炸桥. 如果无法使曹操的点成为多个连通图,则输出-1. 思路: 就是用tarjan算法算出桥的数量,再比较

ZOJ 1588 Burning Bridges (tarjan求割边)

题目链接 题意 : N个点M条边,允许有重边,让你求出割边的数目以及每条割边的编号(编号是输入顺序从1到M). 思路 :tarjan求割边,对于除重边以为中生成树的边(u,v),若满足dfn[u] < low[v],则边(u,v)是割边. 1 #include <iostream> 2 #include <cstdio> 3 #include <cstring> 4 #include <algorithm> 5 6 using namespace st