题目链接:https://vjudge.net/problem/HDU-4612

题目:一个大地图,给定若干个连通图,每个连通图中有若干个桥,你可以在任意某个连通图的

任意两个点添加一条边,问,添加一条边后,大地图中最少剩下几个桥。

思路:tarjan缩点,重构图,对每个新图跑两次dfs求出树的直径,取所有新图的直径max,

答案就是  大地图总桥数 - max(树的直径)。

 #include <iostream>
#include <cstdio>
#include <algorithm>
#include <queue>
#include <vector>
using namespace std;
#define pb push_back const int N = (int)2e5+;
const int M = (int)1e6+;
int n,m,bridge,tot,tim,top,scc;
int head[N],dfn[N],low[N],s[N],scc_no[N],vis[N];
struct node{
int to;
int nxt;
}e[M << ];
vector<int> g[N];//新图
vector<int> poi;//存单个连通图包含的点 void init(){
for(int i = ; i <= n; ++i){
head[i] = -;
dfn[i] = ;
g[i].clear();
}
bridge = tot = tim = top = scc = ;
} inline void add(int u,int v){
e[tot].to = v;
e[tot].nxt = head[u];
head[u] = tot++;
} //tarjan缩点
void tarjan(int now,int pre){
poi.pb(now);//存下这个连通图包含的点
dfn[now] = low[now] = ++tim;
s[top++] = now;
int to,pre_cnt = ;
for(int o = head[now]; ~o; o = e[o].nxt){
to = e[o].to;
if(to == pre && pre_cnt == ) { pre_cnt = ; continue; }
if(!dfn[to]){
tarjan(to,now);
low[now] = min(low[now],low[to]);
if(dfn[now] < low[to]) ++bridge;
}else low[now] = min(low[now],dfn[to]);
} if(dfn[now] == low[now]){
int x;
++scc;
do{
x = s[--top];
scc_no[x] = scc;
}while(now != x);
}
} //对poi中的那些点新建一个图
void rebuild(){
int to,now;
for(int i = ; i < (int)poi.size(); ++i){
now = poi[i];
for(int o = head[now]; ~o; o = e[o].nxt){
to = e[o].to;
if(scc_no[now] == scc_no[to]) continue;
g[scc_no[now]].pb(scc_no[to]); g[scc_no[to]].pb(scc_no[now]);
}
}
} void dfs(int now,int pre,int deep){
vis[now] = deep;
int to;
for(int i = ; i < (int)g[now].size(); ++i){
to = g[now][i];
if(to == pre || to == now || vis[to]) continue;
dfs(to,now,deep+);
}
} void test01(){
for(int i = ; i <= n; ++i)
printf("%d 属于scc = %d\n",i,scc_no[i]);
} void solve(){ int max_deep = ,_deep = ;
int u;
for(int i = ; i <= n; ++i){
if(!dfn[i]){
poi.clear();//清空上个连通图中的点
tarjan(i,i);
rebuild();//poi中的点重建图
dfs(poi[],poi[],);
_deep = ;
for(int i = ; i < poi.size(); ++i){
if(_deep < vis[poi[i]]){ _deep = vis[poi[i]]; u = poi[i]; }
vis[poi[i]] = ; //这里别忘了初始化 不然下次dfs会出错
}
dfs(u,u,);
for(int i = ; i < poi.size(); ++i){
_deep = max(_deep,vis[poi[i]]);
vis[poi[i]] = ;//这里别忘了初始化 不然下组数据的dfs会出错
}
max_deep = max(max_deep,_deep);//对每个连通图跑两次dfs求树的直径,取max
}
}
// cout << bridge << " " << max_deep << endl;
printf("%d\n",bridge - max_deep +);
} int main(){ int u,v;
while(~scanf("%d%d",&n,&m) && (n+m)){
init();
for(int i = ; i < m; ++i){
scanf("%d%d",&u,&v);
add(u,v); add(v,u);
}
solve();
} return ;
}

4 4
1 2
1 3
1 4
2 3

11 12
1 2
1 3
3 2
3 4
4 5
4 6
6 7
7 8
7 9
8 9
8 11
9 10

6 7
1 2
1 3
2 3
3 4
4 5
4 6
5 6

8 6
1 2
1 3
1 4
1 5
7 6
6 8

最新文章

  1. Hadoop op 1)
  2. How to Install The Alpha Control Packages.
  3. 错误 1 error LNK1123: 转换到 COFF 期间失败: 文件无效或损坏
  4. Github快速入门手册
  5. VC6.0 error LNK2001: unresolved external symbol _main解决办法
  6. 有N个大小不等的自然数(1--N),请将它们由小到大排序。要求程序算法:时间复杂度为O(n),空间复杂度为O(1)。
  7. JS调用iframe方式实现Web区域打印页面内容
  8. 引用jquery框架后出错
  9. AdminLTE的使用
  10. 输入3行字符串/定义flag/while/字符串后要加空格符
  11. Python编程Day6——元组类型、字典类型、集合
  12. MSSQL 如何采用sql语句 获取建表字段说明、字段备注、字段类型、字段长度
  13. mysql安装设置mysql字符集utf8及修改密码
  14. 【实战】Docker 入门实战一:ubuntu 和 centos 安装Docker
  15. Nginx 自动补全url地址补全最后的斜线
  16. import的使用
  17. RabbitMQ系列教程之六:远程过程调用(RPC)(转载)
  18. google image
  19. linux shell 脚本攻略学习18--grep命令详解
  20. 最全的CSS浏览器兼容问题【CSS技巧 】

热门文章

  1. 安装 Xen
  2. 个人第四次作业:Alpha项目测试
  3. svg微信公众号推文实现点击显示答案
  4. SpringBoot + Mybatis 和ssm 使用数据库的区别
  5. Django自动化测试平台项目案例
  6. SpringBoot整合ActiveMQ和开启持久化
  7. 目标检测之单步检测(Single Shot detectors)
  8. POJ_2479_DP
  9. To be contine ,NW NMM backup sqlserver failed.
  10. C++解析Json,使用JsonCpp读写Json数据