题意:有一些联通的地方,如果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 强连通的更多相关文章

  1. HDU 4612——Warm up——————【边双连通分量、树的直径】

    Warm up Time Limit:5000MS     Memory Limit:65535KB     64bit IO Format:%I64d & %I64u Submit Stat ...

  2. Hdu 4612 Warm up (双连通分支+树的直径)

    题目链接: Hdu 4612 Warm up 题目描述: 给一个无向连通图,问加上一条边后,桥的数目最少会有几个? 解题思路: 题目描述很清楚,题目也很裸,就是一眼看穿怎么做的,先求出来双连通分量,然 ...

  3. Hdu 1269 强连通判定

    题目链接 迷宫城堡 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total S ...

  4. HDU 3072 (强连通分量)

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=3072 题目大意:为一个有向连通图加边.使得整个图全连通,有重边出现. 解题思路: 先用Tarjan把 ...

  5. hdu 4612 Warm up

    http://acm.hdu.edu.cn/showproblem.php?pid=4612 将原图进行缩点 变成一个树 树上每条边都是一个桥 然后加一条边要加在树的直径两端才最优 代码: #incl ...

  6. 【HDU 4612 Warm up】BCC 树的直径

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=4612 题意:一个包含n个节点m条边的无向连通图(无自环,可能有重边).求添加一条边后最少剩余的桥的数 ...

  7. hdu 4685(强连通分量+二分图)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4685 题意:n个王子和m个公主,王子只能和他喜欢的公主结婚,公主可以和所有的王子结婚,输出所有王子可能 ...

  8. (求树的直径)Warm up -- HDU -- 4612

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4612 给一个无向图, 加上一条边后,求桥至少有几个: 那我们加的那条边的两个顶点u,v:一定是u,v之 ...

  9. HDU 4612 Warm up tarjan 树的直径

    Warm up 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=4612 Description N planets are connected by ...

随机推荐

  1. NOIP2000乘积最大[序列DP]

    题目描述 今年是国际数学联盟确定的“2000――世界数学年”,又恰逢我国著名数学家华罗庚先生诞辰90周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学智力竞赛的活动,你的一个好朋友XZ也有幸得 ...

  2. 批处理文件指定jre路径启动java桌面应用程序

    应用场景: 我开发了一个应用程序,并连同jre一起刻成光盘,提供给用户,用户直接双击批处理文件即可运行,而不需要自己额外装jre. 目录组织结构如下: client |-images |-jre |- ...

  3. JProfiler学习笔记

    JProfiler学习笔记   一.安装JProfiler        从http://www.ej-technologies.com/下载5.1.2并申请试用序列号   二.主要功能简介 1.内存 ...

  4. java 27 - 9 反射之 动态代理的概述和实现

    代理:本来应该自己做的事情,却请了别人来做,被请的人就是代理对象. 举例:春季回家买票让人代买 动态代理: 在程序运行过程中产生的这个对象 而程序运行过程中产生对象其实就是我们刚才反射讲解的内容,所以 ...

  5. 技术专题—Python黑客【优质内容聚合贴】

    作者:坏蛋链接:https://zhuanlan.zhihu.com/p/24645819来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处. 一.前言 本着知识分享,聚合优 ...

  6. class 函数

    cocos2d-x 3.3 lua 版本的class函数用法: local FightScene = class("FightScene", function() return c ...

  7. Construct Binary Tree from Preorder and Inorder Traversal

    Construct Binary Tree from Preorder and Inorder Traversal Given preorder and inorder traversal of a ...

  8. 详细学习ORACLE JOBS

    一点一点学习jobs的各个方面比较长,比较烦,但是应该看完后会对jobs比较好的应用 一.学习准备 开始dbms_job学习前,先认识一个参数job_queue_processes a.job_que ...

  9. opencv3-core之基本操作

    这一篇打算将core部分的例子说完,这都是基于<opencv2.4.9tutorial.pdf>中的core部分,其实这些例子后期都很稳定的,也就是说就算是2.3.1和2.4.10 ,这几 ...

  10. [MCSM]随机搜索和EM算法

    1. 概述 本节将介绍两类问题的不同解决方案.其一是通过随机的搜索算法对某一函数的取值进行比较,求取最大/最小值的过程:其二则和积分类似,是使得某一函数被最优化,这一部分内容的代表算法是EM算法.(书 ...