【HDU4612】 双连通分量求桥
题目链接: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】 双连通分量求桥的更多相关文章
- hdoj 4612 Warm up【双连通分量求桥&&缩点建新图求树的直径】
Warm up Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65535/65535 K (Java/Others)Total Su ...
- zoj 2588 Burning Bridges【双连通分量求桥输出桥的编号】
Burning Bridges Time Limit: 5 Seconds Memory Limit: 32768 KB Ferry Kingdom is a nice little cou ...
- hdoj 4738 Caocao's Bridges【双连通分量求桥】
Caocao's Bridges Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
- 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) ...
- hdu 3849 (双联通求桥)
一道简单的双联通求桥的题目,,数据时字符串,,map用的不熟练啊,,,,,,,,,,,,, #include <iostream> #include <cstring> #in ...
- hdu 4738 (双联通求桥)
2013 ACM/ICPC Asia Regional Hangzhou Online 题目大意:有n个岛,曹操在一些岛之间建了一些桥,每个桥上有一些士兵把守,周瑜只有一个炸弹只能炸掉一个桥,炸弹需要 ...
- POJ 3352 Road Construction(边双连通分量,桥,tarjan)
题解转自http://blog.csdn.net/lyy289065406/article/details/6762370 文中部分思路或定义模糊,重写的红色部分为修改过的. 大致题意: 某个企业 ...
- fzu2181(点的双连通分量+求奇环)
求出每个点双连通分量,如果在一个点双连通分量中有奇环,则这个分量每个点都在一个奇环中. 关键是要知道怎么求点双连通分量以及点双连通的性质. fzu2181 http://acm.fzu.edu.cn ...
- 2013杭州网赛 1001 hdu 4738 Caocao's Bridges(双连通分量割边/桥)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4738 题意:有n座岛和m条桥,每条桥上有w个兵守着,现在要派不少于守桥的士兵数的人去炸桥,只能炸一条桥 ...
随机推荐
- 【POI】使用POI处理xlsx的cell中的超链接 和 插入图片 和 设置打印区域
使用POI对xlsx中插入超链接和 插入图片 package com.it.poiTest; import java.awt.image.BufferedImage; import java.io.B ...
- Android自动化压力测试之Monkey Test (三)
Monkey 是什么? Monkey是Google提供的一个用于稳定性与压力测试的命令行工具. Monkey路径: 路径:/System/framework/monkey.jar 启动脚本路径:/sy ...
- strcmp函数使用总结
Action() { /********************************* * Author:旺仔 * object:strcmp * date:2015-12-09 * fuc:我输 ...
- ffmpeg处理RTMP流媒体的命令 发送流媒体的命令(UDP,RTP,RTMP)
将文件当做直播送至live ffmpeg -re -i localFile.mp4 -c copy -f flv rtmp://server/live/streamName re限制输出速率,按照 ...
- Swift3.0语言教程分割字符串与截取字符串
Swift3.0语言教程分割字符串与截取字符串 Swift3.0语言教程分割字符串 如果想要快速的创建一个数组,我们可以将字符串进行分割,分割后的内容将会生成一个数组.在NSString中有两个分割字 ...
- javascript关键字和保留字
1 关键字breakcasecatchcontinuedefaultdeletedoelsefinallyforfunctionifininstanceofnewreturnswitchthisthr ...
- zoj 3469 Food Delivery 区间dp + 提前计算费用
Time Limit: 2 Seconds Memory Limit: 65536 KB When we are focusing on solving problems, we usual ...
- AngularJS 包含HTML文件
类似于python tornado的include方法,同样是可以在一个html文件中加载另外一个html文件,这样可以不用重复的写一些几乎不改变的代码. 首先创建两个文件,然后代码如下: <! ...
- 【转】win8.1下安装ubuntu
参考:http://jingyan.baidu.com/article/14bd256e0ca52ebb6d26129c.html Ubuntu 系统是一款优秀的.基于GNU/Linux 的平台的桌面 ...
- Android系统中的6种模式
Android系统中的6种模式 1:一般启动模式(normal mode): 功能是正常启动手机,方法为关机状态下按电源键启动. 2:安全模式(safe mode): 此模式和正常启动一样 ...