/*
对于边双连通分支,求法更为简单。 仅仅需在求出全部的桥以后,把桥边删除。\
原图变成了多个连通块,则每一个连通块就是一个边双连通分支。 桥不属于不论什么
一个边双连通分支,其余的边和每一个顶点都属于且仅仅属于一个边双连通分支。
一个有桥的连通图,怎样把它通过加边变成边双连通图?方法为首先求出全部的桥,
然后删除这些桥边,剩下的每一个连通块都是一个双连通子图。把每一个双连通子图收缩为一个顶点,
再把桥边加回来,最后的这个图一定是一棵树。边连通度为1。
统计出树中度为1的节点的个数,即为叶节点的个数,记为leaf。 则至少在树上加入(leaf+1)/2条边。
就能使树达到边二连通。所以至少加入的边数就是(leaf+1)/2。详细方法为,首先把两个近期公共祖先最远
的两个叶节点之间连接一条边,这样能够把这两个点到祖先的路径上全部点收缩到一起。
由于一个形成的环一定是双连通的。然后再找两个近期公共祖先最远的两个叶节点,这样一对一对找完,
恰好是(leaf+1)/2次。把全部点收缩到了一起。
*/
/*
(1) n 为顶点数, 标号从 1 開始
(2) c 为原图的邻接表, g 为 E_BCC 图的邻接表
(3) num[u] 表示原图中的点 u 属于新图中的第 num[u] 个 E_BCC
(4) edge[] 存储全部的桥
(5) 注意 pool[M] 要开得足够大以容得下新旧两个图中全部的边
(6) E_BCC 图中去掉了自环 ( 显然不存在多重边 )
*/
#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;
const int N = 11115;
const int M = 2000005; struct List {
int v, id;
List *next;
} pool[M], *c[N], *g[N], *pp;
//c 为原图的邻接表, g 为 E_BCC 图的邻接表
//注意 pool[M] 要开得足够大以容得下新旧两个图中全部的边
inline void add_edge(int u, int v, int id, List *c[])
{
pp->v = v;
pp->id = id;
pp->next = c[u];
c[u] = pp ++;
} struct Edge {
int u, v;
} edge[M];
//edge[] 存储全部的桥,u,v为桥的两个顶点
int n, m, label, tot, top;
int low[N], dfn[N], num[N], stack[N];
bool eflag[M];
//label时间戳,tot连通块数
//dfn用来保存时间戳(次序)编号,low保存顶点i或i的子树最早的次序编号
//num[u] 表示原图中的点 u 属于新图中的第 num[u] 个 E_BCC
void E_BCC_VISIT(int u)
{
low[u] = dfn[u] = label ++;
stack[++ top] = u;
for(List *p = c[u]; p; p = p->next) {
int v = p->v;
if(eflag[p->id]) continue;
eflag[p->id] = true;
//if(dfn[v]) { low[u] <?= dfn[v]; continue; }
if(dfn[v]){
if(low[u] > dfn[v]) low[u] = dfn[v];
continue;
}
E_BCC_VISIT(v);
//low[u] <?= low[v];
if(low[u] > low[v]) low[u]=low[v];
if(low[v] > dfn[u]) {
edge[m].u = u;//第m条桥的两个顶点u,v
edge[m ++].v = v;
++ tot;
do {
num[stack[top]] = tot;
} while( stack[top --] != v );
}
}
}
void E_BCC()
{
int i;
tot = 0;
m = 0;/////
for(i = 1; i <= n; ++ i) dfn[i] = 0, num[i] = -1;
for(i = 0; i < m; ++ i) eflag[i] = false;
for(i = 1; i <= n; ++ i)
if(dfn[i] == 0) {
label = 1;
top = -1;
E_BCC_VISIT(i);
++ tot;
while( top >= 0 ) {
num[stack[top]] = tot;
-- top;
}
}
for(i = 1; i <= tot; ++ i) g[i] = NULL;
//for(i = 1; i <= n; ++ i) {//缩点,这题用不着
// int u = num[i];//u为一个双连通分量
//for(List *p = c[i]; p; p = p->next) {
// int v = num[p->v];//v是还有一个双连通分量
//if(u != v) add_edge(u, v, 0, g);//在两个分量间建一条边
//}
//}
} int main()
{
int i, j, k;
while( scanf("%d %d", &n, &m) == 2 ) {
for(i = 1; i <= n; ++ i) c[i] = NULL;
pp = pool;
for(k = 0; k < m; ++ k) {
scanf("%d %d", &i, &j);
add_edge(i, j, k, c);
add_edge(j, i, k, c);
}
E_BCC();
if(m == 0){cout<<0<<endl; continue;}
int du[N]={0};
for(int i=0;i<m;i++){//桥即为联通块的之间的边,这里处理伪缩点
//cout<<num[edge[i].u]<<' '<<num[edge[i].v]<<endl;
du[num[edge[i].u]]++;//要用num[]映射到连通块编号上计算联通块的度
du[num[edge[i].v]]++;
}
int leaf=0;//树叶
//cout<<tot<<endl<<m<<endl;
for(int i=1;i<=tot;i++) if(du[i]==1)leaf++;
cout<<(leaf+1)/2<<endl;
//for(int i=0;i<=m;i++)printf("num[%d]:%d\n",i,num[i]);
//printf("tot:%d m:%d\n",tot,m);
}
return 0;
}

poj3352Road Construction 边双连通+伪缩点的更多相关文章

  1. POJ3352Road Construction(构造双连通图)sdut2506完美网络

    构造双连通图:一个有桥的连通图,如何把它通过加边变成边双连通图? 一个有桥的连通图,如何把它通过加边变成边双连通图?方法为首先求出所有的桥,然后删除这些桥边,剩下的每个连通块都是一个双连通子图.把每个 ...

  2. HDU4612Warm up 边双连通 Tarjan缩点

    N planets are connected by M bidirectional channels that allow instant transportation. It's always p ...

  3. 图论--边双连通V-DCC缩点

    // tarjan算法求无向图的割点.点双连通分量并缩点 #include<iostream> #include<cstdio> #include<cstring> ...

  4. 图论--双连通E-DCC缩点模板

    // tarjan算法求无向图的桥.边双连通分量并缩点 #include<iostream> #include<cstdio> #include<cstring> ...

  5. POJ 3177 Redundant Paths POJ 3352 Road Construction(双连接)

    POJ 3177 Redundant Paths POJ 3352 Road Construction 题目链接 题意:两题一样的.一份代码能交.给定一个连通无向图,问加几条边能使得图变成一个双连通图 ...

  6. POJ-3352 Road Construction,tarjan缩点求边双连通!

    Road Construction 本来不想做这个题,下午总结的时候发现自己花了一周的时间学连通图却连什么是边双连通不清楚,于是百度了一下相关内容,原来就是一个点到另一个至少有两条不同的路. 题意:给 ...

  7. poj 3352 Road Construction【边双连通求最少加多少条边使图双连通&&缩点】

    Road Construction Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 10141   Accepted: 503 ...

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

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

  9. POJ 3177 Redundant Paths (边双连通+缩点)

    <题目链接> <转载于 >>>  > 题目大意: 有n个牧场,Bessie 要从一个牧场到另一个牧场,要求至少要有2条独立的路可以走.现已有m条路,求至少要新 ...

随机推荐

  1. 2016.03.10,英语,《Vocabulary Builder》Unit 05

    mal: means bad. malpractice [ˌmæl'præktɪs] n. 失职, 行为不当; malady ['mælədi] n. 病, 疾病, 弊病; malodorous [ˌ ...

  2. 客户现场调试(连接oracle数据库)

    1.System.Data.OracleClient 需要 Oracle 客户端软件 8.1.7 或更高版本 http://blog.csdn.net/yucaoye/article/details/ ...

  3. C# 特性(Attribute)

    C# 特性(Attribute) 特性(Attribute)是用于在运行时传递程序中各种元素(比如类.方法.结构.枚举.组件等)的行为信息的声明性标签.您可以通过使用特性向程序添加声明性信息.一个声明 ...

  4. JQuery学习系列篇(二)

    1.事件切换函数 hover([over],out); over鼠标移动到元素上要触发的函数,out鼠标移出元素要触发的函数. 2.togger 如果元素是可见的,切换为隐藏的:如果元素是隐藏的,切换 ...

  5. EntityFramework使用及优化

    1. 简介 ORM框架:Object Relation Mapping,用操作对象的方式来操作数据库 其它框架:Dapper.NHibernate,首推EF,微软官方的. EF底层还是ADO.NET实 ...

  6. SpringBoot(九) ElasticSearch 全文检索

    ElasticSearch ​ ElasticSearch是一个基于Lucene的搜索服务器.它提供了一个分布式多用户能力的全文搜索引擎,基于RESTful web接口.Elasticsearch是用 ...

  7. JavaScript DOM编程艺术(第2版)学习笔记1(1~4章)

    第一章 一些基本概念 HTML(超文本标记语言),构建网页的静态结构,由一系列的DOM组成: CSS(层叠样式表),给网页各部分结构添加样式: JavaScript,通过获取DOM给静态结构加上动作, ...

  8. jmeter的认识——线程组的认识

    名称:可以给线程组设置一个个性化的命名 注释:可以对线程组添加备注以标记 在取样器错误后要执行的动作:就是在错误之后要如何执行,可选继续执行后续的.停止执行等. 线程数:就是需要设置多少线程执行测试. ...

  9. CSS3背景 制作导航菜单综合练习题

    CSS3背景 制作导航菜单综合练习题 小伙伴们,根据所学知识,使用CSS3实现下图的导航菜单效果 任务 1.制作导航圆角 提示:使用border-radius实现圆角 2.制作导航立体风格 提示:使用 ...

  10. JEE Spring-boot 简单的ioc写法。

    什么是ioc,就是你可能会有一些生活必需品,这些东西你必须要用才能存活.但是你不是每天都回去买,去哪一家点去买.而这些用品会一直放在哪里,每一个商店就是一个容器,包裹着这些物品. 创建ioc项目,首先 ...