hdu 4612 强连通
题意:有一些联通的地方,如果2点间只有一条路径,这样的边叫做桥,现在让你添加一个桥,使最后的桥最少,问最少的桥使多少?
先求一次强连通分量,然后图就分成了几个块,将这几个块看做点,求出总共有多少条重建图中的边,然后再求出最长的边,这样答案就是总共
边数 - 最长的边的长度。原来的标记手法过了几题,但是做这题有点问题,用了另一个手法。
#pragma comment(linker, "/STACK:102400000,102400000")
#include<map>
#include<queue>
#include<stack>
#include<cmath>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define INF 1000000001
#define ll __int64
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
using namespace std;
const int MAXN = ;
struct node
{
int to;
int next;
int vis;
}edge[ * ],a[ * ];
struct fnode
{
int x;
int val;
};
int n,m,ind,pre[MAXN],ins[MAXN],low[MAXN],dfn[MAXN],vis[MAXN],num[MAXN],fpre[MAXN];
int fp,len,p;
stack<int>s;
void Init()
{
fp = ;
ind = ;
while(!s.empty())s.pop();
memset(num,,sizeof(num));
memset(ins,,sizeof(ins));
memset(pre,-,sizeof(pre));
memset(vis,,sizeof(vis));
memset(dfn,,sizeof(dfn));
memset(low,,sizeof(low));
}
void add(int x,int y)
{
edge[ind].to = y;
edge[ind].vis = ;
edge[ind].next = pre[x];
pre[x] = ind ++;
}
void dfs(int rt,int k,int fa)
{
s.push(rt);
vis[rt] = ins[rt] = ;
low[rt] = dfn[rt] = k;
for(int i = pre[rt]; i != -; i = edge[i].next){
int t = edge[i].to;
if(edge[i].vis)continue;
edge[i].vis = edge[i^].vis = ;
//原来判断 fa != t 但是有重边的情况这种标记不行。
if(!dfn[t]){
dfs(t,k+,rt);
low[rt] = min(low[rt],low[t]);
}
else{
low[rt] = min(dfn[t],low[rt]);
}
}
if(low[rt] == dfn[rt]){
while(!s.empty()){
int temp = s.top();
s.pop();
num[temp] = fp;
if(temp == rt)break;
}
fp ++;
}
}
void add_b(int x,int y)
{
a[ind].to = y;
a[ind].next = fpre[x];
fpre[x] = ind ++;
}
void build(int rt)
{
vis[rt] = ;
for(int i = pre[rt]; i != -; i=edge[i].next){
int t = edge[i].to;
if(!vis[t]){
if(num[t] != num[rt]){
add_b(num[t],num[rt]);
add_b(num[rt],num[t]);
}
build(t);
}
}
}
void bfs(int rt)
{
queue<fnode>q;
fnode temp;
temp.val = ;
temp.x = rt;
vis[rt] = ;
q.push(temp);
while(!q.empty()){
temp = q.front();
q.pop();
for(int i = fpre[temp.x]; i != -; i = a[i].next){
int t = a[i].to;
if(!vis[t]){
fnode ft;
ft.val = temp.val + ;
ft.x = t;
vis[t] = ;
q.push(ft);
if(ft.val > len){
len = ft.val;
p = ft.x;
}
}
}
}
}
int main()
{
while(~scanf("%d%d",&n,&m)){
if(!n && !m)break;
Init();
for(int i = ; i <= m; i++){
int x,y;
scanf("%d%d",&x,&y);
add(x,y);
add(y,x);
}
for(int i = ; i <= n; i++){
if(!dfn[i]){
dfs(i,,-);
}
}
memset(vis,,sizeof(vis));
ind = ;
memset(fpre,-,sizeof(fpre));
for(int i = ; i <= n; i++){
if(!vis[i]){
build(i);
}
}
len = ;
p = ;
memset(vis,,sizeof(vis));
bfs();
len = ;
memset(vis,,sizeof(vis));
bfs(p);
fp -= ;
int ans = fp - - len;
//cout<<"fp = "<<fp<<" len = "<<len<<endl;
cout<<(ans > ? ans : )<<endl;
}
return ;
}
hdu 4612 强连通的更多相关文章
- HDU 4612——Warm up——————【边双连通分量、树的直径】
Warm up Time Limit:5000MS Memory Limit:65535KB 64bit IO Format:%I64d & %I64u Submit Stat ...
- Hdu 4612 Warm up (双连通分支+树的直径)
题目链接: Hdu 4612 Warm up 题目描述: 给一个无向连通图,问加上一条边后,桥的数目最少会有几个? 解题思路: 题目描述很清楚,题目也很裸,就是一眼看穿怎么做的,先求出来双连通分量,然 ...
- Hdu 1269 强连通判定
题目链接 迷宫城堡 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total S ...
- HDU 3072 (强连通分量)
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=3072 题目大意:为一个有向连通图加边.使得整个图全连通,有重边出现. 解题思路: 先用Tarjan把 ...
- hdu 4612 Warm up
http://acm.hdu.edu.cn/showproblem.php?pid=4612 将原图进行缩点 变成一个树 树上每条边都是一个桥 然后加一条边要加在树的直径两端才最优 代码: #incl ...
- 【HDU 4612 Warm up】BCC 树的直径
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=4612 题意:一个包含n个节点m条边的无向连通图(无自环,可能有重边).求添加一条边后最少剩余的桥的数 ...
- hdu 4685(强连通分量+二分图)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4685 题意:n个王子和m个公主,王子只能和他喜欢的公主结婚,公主可以和所有的王子结婚,输出所有王子可能 ...
- (求树的直径)Warm up -- HDU -- 4612
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4612 给一个无向图, 加上一条边后,求桥至少有几个: 那我们加的那条边的两个顶点u,v:一定是u,v之 ...
- HDU 4612 Warm up tarjan 树的直径
Warm up 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=4612 Description N planets are connected by ...
随机推荐
- Codeforces 715A. Plus and Square Root[数学构造]
A. Plus and Square Root time limit per test 2 seconds memory limit per test 256 megabytes input stan ...
- Vijos1680距离/openjudge2988计算字符串的距离[DP]
描述 设有字符串X,我们称在X的头尾及中间插入任意多个空格后构成的新字符串为X的扩展串,如字符串X为”abcbcd”,则字符串“abcb_c_”,“_a_bcbcd_”和“abcb_c_”都是X的扩展 ...
- AC日记——将字符串中的小写字母换成大写字母 openjudge 1.7 13
13:将字符串中的小写字母转换成大写字母 总时间限制: 1000ms 内存限制: 65536kB 描述 给定一个字符串,将其中所有的小写字母转换成大写字母. 输入 输入一行,包含一个字符串(长度不 ...
- bzoj1036 [ZJOI2008]树的统计Count
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MB Submit: 12646 Solved: 5085 [Subm ...
- Android保存ArrayList至SharedPreferences
保存ArrayList至SharedPreferences 其中ArrayList中每个元素为String List<String> environmentList = new Array ...
- C# — FileHandler
学会使用OpenFileDialog和SaveFileDialog控件浏览和选择文件.使用System.IO.File和System.IO.Directory的对象来操纵文件系统(文件和目录). 在F ...
- Mysql优化系列(1)--Innodb引擎下mysql自身配置优化
1.简单介绍InnoDB给MySQL提供了具有提交,回滚和崩溃恢复能力的事务安全(ACID兼容)存储引擎.InnoDB锁定在行级并且也在SELECT语句提供一个Oracle风格一致的非锁定读.这些特色 ...
- 理解java的finalize
基本预备相关知识 1 java的GC只负责内存相关的清理,所有其它资源的清理必须由程序员手工完成.要不然会引起资源泄露,有可能导致程序崩溃. 2 调用GC并不保证GC实际执行. 3 finalize抛 ...
- Java 基础命名空间
java.lang (提供利用 Java 编程语言进行程序设计的基础类)java.lang.annotation(提供了引用对象类,支持在某种程度上与垃圾回收器之间的交互)java.lang.inst ...
- ASP.NET整理:Cookie,Application,Session,页面生命周期
一.设置Cookie的2种方式 1. Repsonse.Cookie[“名”] = 值; 2. HttpCookie hcCookie = new HttpCookie(“名”,值); h ...