<题目链接>

题目大意:
给出一个连通图,问你在这个连通图上加一条边,使该连通图的桥的数量最小,输出最少的桥的数量。

解题分析:

首先,通过Tarjan缩点,将该图缩成一颗树,树上的每个节点都是一个边双连通分量,树上的每条边都是桥,现在需要挑出两个点,将它们直接相连,这样它们原始路径上所有的桥因为形成了环而全部消失,因此为了使剩下的桥最少,我们需要找到路径上桥最多的两点,又由于缩点后,树的每条边都是桥,所以这里就转化为树上距离两点的最远距离,也就是求树的直径。

下面Tarjan的时候需要注意的是,vis记录的是访问过边的编号不是节点的编号。

 #include <iostream>
 #include <cstdio>
 #include <cstring>
 #include <algorithm>
 #include <queue>
 using namespace std;
 #define mem(a,b) memset(a,b,sizeof(a))
 typedef pair<int, int> pll;
 ;
 ;
 int num, num1, top, cnum,maxdis, pos,cnt;
 int head[N], head1[N], dis[N],instk[N], stk[N], dfn[N], low[N], belong[N];
 ];
 struct Edge{
     int to, next;
 }e[M<<], e1[M<<];

 void add(int u, int v) {
     e[num].to = v, e[num].next = head[u], head[u] = num++;
 }
 void add1(int u, int v) {
     e1[num1].to = v, e1[num1].next = head1[u], head1[u] = num1++;
 }
 void init() {
     num = num1 = cnt = cnum = top = ;
     mem(head,-),mem(head1,-),mem(belong,),mem(instk,);
     mem(vis,),mem(stk,),mem(dfn,),mem(low,),mem(dis,);
 }
 pll edge[M];
 void tarjan(int u){
     dfn[u] = low[u] = ++cnt;
     instk[u] = ;
     stk[++top] = u;
     for(int i = head[u]; ~i; i = e[i].next) {
         int v = e[i].to;
         if(vis[i])continue;
         vis[i] = vis[i^] = ;    //将正、反两边全部标记
         if(!dfn[v]) {
             tarjan(v);
             low[u] = min(low[u], low[v]);
         } else if(instk[v]) {
             low[u] = min(low[u], dfn[v]);
         }
     }
     if(low[u] == dfn[u]) {
         ++cnum;      //cnum为边双连通分量的数量
         int v;
         do{
             v = stk[top--];
             instk[v] = ;
             belong[v] = cnum;     //将该连通块染色
         } while(v != u);
     }
 }
 void bfs(int u){
     queue <int> q;
     q.push(u);
     dis[u] = ;
     mem(vis,);
     vis[u] = ;
     maxdis = , pos = u;
     while(!q.empty()) {
         int u = q.front(); q.pop();
         for(int i = head1[u]; ~i; i = e1[i].next) {
             int v = e1[i].to;
             if(vis[v])continue;
             vis[v] = ;
             dis[v] = dis[u]+;
             if(dis[v]>maxdis) {     //更新树上的最远距离
                 maxdis = dis[v];
                 pos = v;     //并且记录下该点
             }
             q.push(v);
         }
     }
 }
 int main()
 {
     int n, m, x, y;
     while(scanf("%d%d",&n,&m)!=EOF,n||m){
         init();
         ; i<m; i++) {
             scanf("%d%d", &x, &y);
             edge[i].first = x, edge[i].second = y;  //记录下所有的边,用结构体记录也可以
             add(x, y),add(y, x);
         }
         tarjan();
         ;
         ; i<m; i++){    //将缩点后的所有点建图,跑BFS,求树的直径
             int x = edge[i].first, y = edge[i].second;
             if(belong[x]!=belong[y]) {
                 add1(belong[x], belong[y]);
                 add1(belong[y], belong[x]);
                 edgenum++;     //桥的数量+1
             }
         }
         bfs(belong[]);
         bfs(pos);     //求出此时树的直径,即一条路径上所含的最多桥的数量
         int ans = edgenum-maxdis;
         printf("%d\n", ans);
     }
 }

2018-11-07

HDU 4612 Warm up (边双连通分量+缩点+树的直径)的更多相关文章

  1. HDU 4612 Warm up(双连通分量缩点+求树的直径)

    思路:强连通分量缩点,建立一颗新的树,然后求树的最长直径,然后加上一条边能够去掉的桥数,就是直径的长度. 树的直径长度的求法:两次bfs可以求,第一次随便找一个点u,然后进行bfs搜到的最后一个点v, ...

  2. hdu4612 Warm up[边双连通分量缩点+树的直径]

    给你一个连通图,你可以任意加一条边,最小化桥的数目. 添加一条边,发现在边双内是不会减少桥的.只有在边双与边双之间加边才有效.于是,跑一遍边双并缩点,然后就变成一棵树,这样要加一条非树边,路径上的点( ...

  3. HDU 4612 Warm up (边双连通分量+DP最长链)

    [题意]给定一个无向图,问在允许加一条边的情况下,最少的桥的个数 [思路]对图做一遍Tarjan找出桥,把双连通分量缩成一个点,这样原图就成了一棵树,树的每条边都是桥.然后在树中求最长链,这样在两端点 ...

  4. Gym - 100676H H. Capital City (边双连通分量缩点+树的直径)

    https://vjudge.net/problem/Gym-100676H 题意: 给出一个n个城市,城市之间有距离为w的边,现在要选一个中心城市,使得该城市到其余城市的最大距离最短.如果有一些城市 ...

  5. hdoj 4612 Warm up【双连通分量求桥&&缩点建新图求树的直径】

    Warm up Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65535/65535 K (Java/Others)Total Su ...

  6. hdu 4612 Warm up 双连通缩点+树的直径

    首先双连通缩点建立新图(顺带求原图的总的桥数,事实上因为原图是一个强连通图,所以桥就等于缩点后的边) 此时得到的图类似树结构,对于新图求一次直径,也就是最长链. 我们新建的边就一定是连接这条最长链的首 ...

  7. HDU-4612 Warm up 边双连通分量+缩点+最长链

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4612 简单图论题,先求图的边双连通分量,注意,此题有重边(admin还逗比的说没有重边),在用targ ...

  8. hdu4612(双连通缩点+树的直径)

    传送门:Warm up 题意:询问如何加一条边,使得剩下的桥的数目最少,输出数目. 分析:tarjan缩点后,重新建图得到一棵树,树上所有边都为桥,那么找出树的直径两个端点连上,必定减少的桥数量最多, ...

  9. HDU 4612 Warm up(2013多校2 1002 双连通分量)

    Warm up Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65535/65535 K (Java/Others)Total Su ...

随机推荐

  1. 其他 Confluence 6 的 cookies 和备注

    其他 Confluence 的 cookies 针对 Confluence 的功能,我们还使用了其他的一些 cookies 来存储基本的 产品持久性(product presentation).Con ...

  2. Confluence 6 通过 SSL 或 HTTPS 运行 - 修改你 Confluence 的 server.xml 文件

    下一步你需要配置 Confluence 来使用 HTTPS: 编辑 <install-directory>/conf/server.xml 文件. 取消注释下面的行: <Connec ...

  3. Cookie禁用了,Session还能用吗?原因详解

    Cookie与 Session,一般认为是两个独立的东西,Session采用的是在服务器端保持状态的方案,而Cookie采用的是在客户端保持状态的方案.但为什么禁用Cookie就不能得到Session ...

  4. mysql之视图,触发器,事务等。。。

    一.视图 视图是一个虚拟表(非真实存在),其本质是[根据SQL语句获取动态的数据集,并为其命名],用户使用时只需使用[名称]即可获取结果集,可以将该结果集当做表来使用. 使用视图我们可以把查询过程中的 ...

  5. jsp 修饰 Request 及Response

    Servlet API包含4个可修饰的类,用于改变Servlet Request以及Servlet Response.这种修饰允许修改 ServletRequest以及ServletResponse或 ...

  6. HTML&javaSkcript&CSS&jQuery&ajax(二)

    一.HTML 1.标签<a href="http:www.baidu.com">This is a link </a>         <img sr ...

  7. CentOS7+mysql(yum)

    1.现在centos上默认是没有yum源的,yum安装的是 MariaDB.所以我们需要自己先配置yum源.配置yum源步骤如下: 下载yum源:wget 'https://dev.mysql.com ...

  8. MySQL5.7.20报错Access denied for user 'root'@'localhost' (using password: NO)

    在centos6.8上源码安装了MySQL5.7.20,进入mysql的时候报错如下: 解决办法如下: 在mysql的配置文件内加入: vim  /etc/my.cnf skip-grant-tabl ...

  9. 彻头彻尾理解 LinkedHashMap

    HashMap和双向链表合二为一即是LinkedHashMap.所谓LinkedHashMap,其落脚点在HashMap,因此更准确地说,它是一个将所有Entry节点链入一个双向链表的HashMap. ...

  10. Vue 添加外部的时间插件不触发v-model事件更改数据

    使用的jquery日期插件 最终问题是 在选择完成日期后并未激活 oninput事件,所以也没有激活v-model 去改变date 解决思路: 去插件js文件中,在赋值给dom的时候添加模拟输入事件便 ...