题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4612

题目大意:给你一个无向图,问你加一条边后最少还剩下多少多少割边。

解题思路:好水的一道模板题。先缩点变成一颗树,再求树的最长直径,直径两端连一条边就是最优解了。

但是....我WA了一个下午.....没有处理重边。

重边的正确处理方法:只标记已经走过的正反边,而不限制已走过的点。换句话说就是可以经过重边再次走向父亲节点,而不能经过走过边的反向边返回父亲节点。

 #pragma comment(linker, "/STACK:1024000000,1024000000")
#include <iostream>
#include <cstdio>
#include <queue>
#include <algorithm>
#include <cstring>
using namespace std; const int maxn=;
const int maxm=;
int dfn[maxn], low[maxn], head[maxn], stack[maxn], instack[maxn], belong[maxn];
int reach[maxm], next[maxm], sign[maxm];
int visit[maxn];
int top, Index, scnt, edge, pos, bridgenum;
int n, m;
pair<int,int>num[maxm]; struct node
{
int u;
int dis;
int fa;
};
queue<node>q; void init()
{
memset(head,-,sizeof(head));
memset(dfn,,sizeof(dfn));
memset(instack,,sizeof(instack));
Index=top=edge=scnt=bridgenum=;
} void addedge(int u, int v)
{
sign[edge]=, reach[edge]=v, next[edge]=head[u], head[u]=edge++;
sign[edge]=, reach[edge]=u, next[edge]=head[v], head[v]=edge++;
} void tarjan(int u)
{
stack[++top]=u;
dfn[u]=low[u]=++Index;
instack[u]=;
for(int i=head[u]; i>=; i=next[i])
{
if(sign[i]) continue;
sign[i]=, sign[i^]=; ///处理重边,只标记边,而不限制点
int v=reach[i];
if(!dfn[v])
{
tarjan(v), low[u]=min(low[u],low[v]);
if(dfn[u]<low[v]) bridgenum++; ///求桥的数量
}
else if(instack[v]) low[u]=min(low[u],dfn[v]);
}
if(low[u]==dfn[u])
{
int v;
scnt++; ///双连通分量的数量
do
{
v=stack[top--];
instack[v]=;
belong[v]=scnt;
}
while(u!=v);
}
} int bfs(int st)
{
int maxx=;
while(!q.empty()) q.pop();
memset(visit,,sizeof(visit));
node s, p;
s.u=st, s.dis=, s.fa=-;
q.push(s);
visit[s.u]=;
while(!q.empty())
{
p=q.front();
q.pop();
for(int i=head[p.u]; i>=; i=next[i])
{
int v=reach[i];
if(v==p.fa) continue;
s.u=v, s.dis=p.dis+, s.fa=p.u;
if(!visit[s.u])
{
visit[s.u]=, q.push(s);
if(s.dis>maxx)
{
maxx=s.dis;
pos=s.u;
}
}
}
}
return maxx;
} void Solve()
{
memset(head,-,sizeof(head));
edge=;
for(int i=; i<=m; i++)
{
int u=num[i].first, v=num[i].second;
int x=belong[u], y=belong[v];
if(x!=y) addedge(x,y);
}
bfs();
int ans=bfs(pos);
cout << bridgenum-ans <<endl;
} int main()
{
while(cin >> n >> m, n+m)
{
init();
for(int i=; i<=m; i++)
{
int u, v;
scanf("%d%d",&u,&v);
num[i].first=u, num[i].second=v;
addedge(u,v);
}
for(int i=; i<=n; i++)
if(!dfn[i]) tarjan(i);
Solve();
}
return ;
} /*
10 11
1 2
2 3
3 4
2 4
3 6
5 10
5 7
7 8
8 9
7 9
3 5
*/

【HDU4612】 双连通分量求桥的更多相关文章

  1. hdoj 4612 Warm up【双连通分量求桥&&缩点建新图求树的直径】

    Warm up Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65535/65535 K (Java/Others)Total Su ...

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

    Burning Bridges Time Limit: 5 Seconds      Memory Limit: 32768 KB Ferry Kingdom is a nice little cou ...

  3. hdoj 4738 Caocao's Bridges【双连通分量求桥】

    Caocao's Bridges Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  4. hdoj 3849 By Recognizing These Guys, We Find Social Networks Useful【双连通分量求桥&&输出桥&&字符串处理】

    By Recognizing These Guys, We Find Social Networks Useful Time Limit: 2000/1000 MS (Java/Others)     ...

  5. hdu 3849 (双联通求桥)

    一道简单的双联通求桥的题目,,数据时字符串,,map用的不熟练啊,,,,,,,,,,,,, #include <iostream> #include <cstring> #in ...

  6. hdu 4738 (双联通求桥)

    2013 ACM/ICPC Asia Regional Hangzhou Online 题目大意:有n个岛,曹操在一些岛之间建了一些桥,每个桥上有一些士兵把守,周瑜只有一个炸弹只能炸掉一个桥,炸弹需要 ...

  7. POJ 3352 Road Construction(边双连通分量,桥,tarjan)

    题解转自http://blog.csdn.net/lyy289065406/article/details/6762370   文中部分思路或定义模糊,重写的红色部分为修改过的. 大致题意: 某个企业 ...

  8. fzu2181(点的双连通分量+求奇环)

    求出每个点双连通分量,如果在一个点双连通分量中有奇环,则这个分量每个点都在一个奇环中.  关键是要知道怎么求点双连通分量以及点双连通的性质. fzu2181 http://acm.fzu.edu.cn ...

  9. 2013杭州网赛 1001 hdu 4738 Caocao's Bridges(双连通分量割边/桥)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4738 题意:有n座岛和m条桥,每条桥上有w个兵守着,现在要派不少于守桥的士兵数的人去炸桥,只能炸一条桥 ...

随机推荐

  1. Callable接口、Runable接口、Future接口

    1. Callable与Runable区别 Java从发布的第一个版本开始就可以很方便地编写多线程的应用程序,并在设计中引入异步处理.Thread类.Runnable接口和Java内存管理模型使得多线 ...

  2. 软件开发中的完整测试所包括的环节UT、IT、ST、UAT

    软件开发中的完成测试环境所包括的环节包括:UT.IT.ST.UAT UT = Unit Test 单元测试 IT = System Integration Test 集成测试ST = System T ...

  3. caffe安装(linux)

    从官网github下载caffe-master.zip 解压:unzip caffe-master 将Makefile.config.example复制,命名为Makefile.config(如果是C ...

  4. 如何清除windows 用户名及密码

  5. JS(ajax笔记)

    简介:AJAX = Asynchronous JavaScript and XML(异步的 JavaScript 和 XML). AJAX 不是新的编程语言,而是一种使用现有标准的新方法,是与服务器交 ...

  6. 我的c++学习(8)运算符重载和友元

    运算符的重载,实际是一种特殊的函数重载,必须定义一个函数,并告诉C++编译器,当遇到该运算符时就调用此函数来行使运算符功能.这个函数叫做运算符重载函数(常为类的成员函数). 方法与解释 ◆ 1.定义运 ...

  7. jquery ajax 提交信息后等待返回的提示信息

    最简单的方法: http://bbs.csdn.net/topics/390584283?page=1 $('#click').click(function){ $('#data').html('&l ...

  8. 百度地图API使用记录

    用户数据图层的总教程: 就是把用户数据存到LBS云里面,应用从云里面读数据 http://developer.baidu.com/map/jsdevelop-9.htm 上传数据的地方: http:/ ...

  9. curl/wget 测试http请求的响应头信息

    1. wget –debug wget可以使用debug信息来查看信息头,如下: [root@localhost ~]# wget --debug http://192.168.1.101:8080/ ...

  10. BZOJ1036[ZJOI2008]树的统计Count 题解

    题目大意: 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.有一些操作:1.把结点u的权值改为t:2.询问从点u到点v的路径上的节点的最大权值 3.询问从点u到点v的路径上的节点的权值和 ...