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 ...
随机推荐
- NOIP2014pj子矩阵[搜索|DP]
题目描述 给出如下定义: 子矩阵:从一个矩阵当中选取某些行和某些列交叉位置所组成的新矩阵(保持行与列的相对顺序)被称为原矩阵的一个子矩阵. 例如,下面左图中选取第2.4行和第2.4.5列交叉位置的元素 ...
- SQL Server的各种聚合函数
聚合函数是对一组值执行计算并返回单一的值的函数,它经常与SELECT语句的GROUP BY子句一同使用,SQL SERVER 中具体有哪些聚合函数呢?我们来一一看一下: 1. AVG 返回指定组中的平 ...
- python画柱状图并且输出到html文件
import matplotlibmatplotlib.use('Agg')import matplotlib.pyplot as pltfrom Cstring import StringIO y ...
- Cornerstone 哪些错误
1.Unable to connect to a repository at URl.............,The operation could not be completed 说明无法连接的 ...
- linux运维中的命令梳理(四)
----------管理命令---------- ps命令:查看进程 要对系统中进程进行监测控制,查看状态,内存,CPU的使用情况,使用命令:/bin/ps (1) ps :是显示瞬间进程的状态,并不 ...
- AngularJS中的缓存
欢迎大家指导与讨论 : ) 缓存篇 一个缓存就是一个组件,它可以透明地储存数据,以便以后可以更快地服务于请求.多次重复地获取资源可能会导致数据重复,消耗时间.因此缓存适用于变化性不大的一些数据,缓存能 ...
- .net异步编程
现在电脑大部分都是多核心,在处理多线程方便有很大优势,异步调用方法的时候可以立即返回执行其他程序,进行异步编程会让程序运行效率更高. 我也是刚刚关注异步编程方面知识,也有很多不是很理解,所以想向大神请 ...
- .Net简单图片系统-简介
系统简介 最近做了一个简单图片系统,这个系统就是 将上传的的图片保存到系统本地文件系统或者基于fastdfs的分布式文件系统中,在查看图片时会直接请求此系统或者fastdfs的tracker服务器(需 ...
- Linux vmstat命令实战详解
vmstat命令是最常见的Linux/Unix监控工具,可以展现给定时间间隔的服务器的状态值,包括服务器的CPU使用率,内存使用,虚拟内存交换情况,IO读写情况.这个命令是我查看Linux/Unix最 ...
- Spark源码在Eclipse中部署/编译/运行
(1)下载Spark源码 到官方网站下载:Openfire.Spark.Smack,其中Spark只能使用SVN下载,源码的文件夹分别对应Openfire.Spark和Smack. 直接下载Openf ...