poj3177 Redundant Paths 边双连通分量
给一个无向图,问至少加入多少条边能够使图变成双连通图(随意两点之间至少有两条不同的路(边不同))。
图中的双连通分量不用管,所以缩点之后建新的无向无环图。
这样,题目问题等效于,把新图中度数为1的点相互连到图里面形成环
如果这种点有sum个,那么至少须要加入(sum+1)/2 条边。
下面,基本上就是求边双连通分量模板。
#include <iostream>
#include <cstring>
#include <string>
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <vector>
#include <queue>
#include <map>
#define inf 0x3f3f3f3f
#define eps 1e-6
#define ll __int64
using namespace std;
#define N 5010
#define M 10010 struct node
{
int v,next,vis;
},e[M];
int h,head[N];
int belong[N],vis[N],dfn[N],low[N],brig[N][2],nbrig,col,cnt,top,in[N],sta[N];
int n,m; void addedge(int a,int b)
{
e[h].v=b;
e[h].vis=0;
e[h].next=head[a];
head[a]=h++;
} void tarjan(int u)
{
int v;
vis[u]=1;
dfn[u]=low[u]=++cnt;
sta[top++]=u;
for(int i=head[u];i!=-1;i=e[i].next)
{
if(e[i].vis) continue;
e[i].vis=e[i^1].vis=1;
v=e[i].v;
if(vis[v]==1)
low[u]=min(low[u],dfn[v]);
if(!vis[v])
{
tarjan(v);
low[u]=min(low[u],low[v]);
if(low[v]>dfn[u])
{
brig[nbrig][0]=u;
brig[nbrig++][1]=v;
}
}
}
if(low[u]==dfn[u])
{
++col;
do
{
v=sta[--top];
vis[v]=0;
belong[v]=col;
}while(u!=v);
}
} void init()
{
memset(head,-1,sizeof head);
memset(vis,0,sizeof vis);
memset(belong,-1,sizeof belong);
h=h1=cnt=col=nbrig=top=0;
} int main()
{
int i,j,k,a,b;
while(~scanf("%d%d",&n,&m))
{
init();
for(i=0;i<m;i++)
{
scanf("%d%d",&a,&b);
addedge(a,b);
addedge(b,a);
}
tarjan(1);
memset(in,0,sizeof in);//记度数//无向图
for(i=0;i<nbrig;i++)//缩点后 树的边是原图里的桥组成
{
a=brig[i][0];
b=brig[i][1];
if(belong[a]!=belong[b])
in[belong[a]]++,in[belong[b]]++;
}
int sum=0;
for(i=1;i<=col;i++)
if(in[i]==1) sum++;
printf("%d\n",(sum+1)/2);
}
return 0;
}
以下这个模板太龊。。
#include <iostream>
#include <cstring>
#include <string>
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <vector>
#include <queue>
#include <map>
#define inf 0x3f3f3f3f
#define eps 1e-6
#define ll __int64
using namespace std;
#define N 1010
#define M 20010 struct node
{
int v;//下个顶点
node *next;//下个边结点
};
int n,m,r[N],d[N];//d表示缩点后各顶点的度数
node mem[M];int memp;//存储边结点
node *e[N];
int necc;//原图中边双连通分量的个数
int belong[N];
int low[N],dfn[N];
int vis[N];
int brig[M][2],nbrig; void addedge(int i,int j)
{
node *p=&mem[memp++];
p->v=j;
p->next=e[i];
e[i]=p;
} int root(int a)
{
if(a!=r[a])
r[a]=root(r[a]);
return r[a];
} void merge(int a,int b)
{
int ra=root(a);
int rb=root(b);
if(ra!=rb)
r[ra]=rb;
} void dfs(int i,int father,int dth)
{
int j,tofa=0;
node *p;
vis[i]=1;
low[i]=dfn[i]=dth;
for(p=e[i];p!=NULL;p=p->next)
{
j=p->v;
if(vis[j]&&(j!=father||tofa))
low[i]=min(low[i],dfn[j]);
if(!vis[j])
{
dfs(j,i,dth+1);
low[i]=min(low[i],low[j]);
if(low[j]<=dfn[i]) merge(i,j);//i,j在同一个双联通分量
if(low[j]>dfn[i]) //边i,j是桥
{
brig[nbrig][0]=i;
brig[nbrig++][1]=j;
}
}
if(j==father) tofa=1;
}
vis[i]=2;
} int doubleconne()
{
int i,k,ncon=nbrig=0;
dfs(0,-1,1);
for(i=0;i<n;i++)
{
k=root(i);
if(belong[k]==-1) belong[k]=ncon++;
belong[i]=belong[k];
}
return ncon;
} void init()
{
memp=nbrig=0;
memset(e,0,sizeof e);
memset(d,0,sizeof d);
for(int i=0;i<=n;i++)
r[i]=i;
memset(vis,0,sizeof vis);
memset(belong,-1,sizeof belong);
} int main()
{
int i,j,k,a,b;
while(~scanf("%d%d",&n,&m))
{
init();
for(i=0;i<m;i++)
{
scanf("%d%d",&a,&b);
addedge(a-1,b-1);
addedge(b-1,a-1);
}
necc=doubleconne();
for(k=0;k<nbrig;k++)
{
i=brig[k][0];
j=brig[k][1];
d[belong[i]]++;
d[belong[j]]++;
}
int cnt=0;//收缩后叶子结点的个数
for(i=0;i<necc;i++)
if(d[i]==1) cnt++;
printf("%d\n",(cnt+1)/2);
}
return 0;
}
poj3177 Redundant Paths 边双连通分量的更多相关文章
- POJ3177 Redundant Paths【双连通分量】
题意: 有F个牧场,1<=F<=5000,现在一个牧群经常需要从一个牧场迁移到另一个牧场.奶牛们已经厌烦老是走同一条路,所以有必要再新修几条路,这样它们从一个牧场迁移到另一个牧场时总是可以 ...
- POJ3177 Redundant Paths —— 边双联通分量 + 缩点
题目链接:http://poj.org/problem?id=3177 Redundant Paths Time Limit: 1000MS Memory Limit: 65536K Total ...
- [POJ3177]Redundant Paths(双联通)
在看了春晚小彩旗的E技能(旋转)后就一直在lol……额抽点时间撸一题吧…… Redundant Paths Time Limit: 1000MS Memory Limit: 65536K Tota ...
- [POJ3177]Redundant Paths(双连通图,割边,桥,重边)
题目链接:http://poj.org/problem?id=3177 和上一题一样,只是有重边. 如何解决重边的问题? 1. 构造图G时把重边也考虑进来,然后在划分边双连通分量时先把桥删去,再划分 ...
- poj 3177 Redundant Paths(边双连通分量+缩点)
链接:http://poj.org/problem?id=3177 题意:有n个牧场,Bessie 要从一个牧场到另一个牧场,要求至少要有2条独立的路可以走.现已有m条路,求至少要新建多少条路,使得任 ...
- POJ3177 Redundant Paths 双连通分量
Redundant Paths Description In order to get from one of the F (1 <= F <= 5,000) grazing fields ...
- POJ3177 Redundant Paths(边双连通分量+缩点)
题目大概是给一个无向连通图,问最少加几条边,使图的任意两点都至少有两条边不重复路径. 如果一个图是边双连通图,即不存在割边,那么任何两个点都满足至少有两条边不重复路径,因为假设有重复边那这条边一定就是 ...
- poj3352 Road Construction & poj3177 Redundant Paths (边双连通分量)题解
题意:有n个点,m条路,问你最少加几条边,让整个图变成边双连通分量. 思路:缩点后变成一颗树,最少加边 = (度为1的点 + 1)/ 2.3177有重边,如果出现重边,用并查集合并两个端点所在的缩点后 ...
- POJ3177 Redundant Paths 图的边双连通分量
题目大意:问一个图至少加多少边能使该图的边双连通分量成为它本身. 图的边双连通分量为极大的不存在割边的子图.图的边双连通分量之间由割边连接.求法如下: 求出图的割边 在每个边双连通分量内Dfs,标记每 ...
随机推荐
- 反编译apk 修改 合成
韩梦飞沙 韩亚飞 313134555@qq.com yue31313 han_meng_fei_sha 反编译apk帮助文档 准备工具 dex2jar(dex转换jar工具),下载地址: ht ...
- Spring和依赖注入的价值
javaeye上看到有帖子,置疑spring和依赖注入的价值,回复内容整理如下: 依赖注入对设计有利,而spring则促进了依赖注入的使用. 如果业务处理类,它所使用的倚赖,都是依靠在这个类内部实现或 ...
- codevs 1014 装箱问题 2001年NOIP全国联赛普及组
题目描述 Description 有一个箱子容量为V(正整数,0<=V<=20000),同时有n个物品(0<n<=30),每个物品有一个体积(正整数). 要求n个物品中,任取若 ...
- 【常见加密方法】Base64编码&Tea算法简介
Base64编码 [Base64编码是什么] Base64是一种基于64个可打印字符来表示二进制数据的表示方法. ——维基百科 Base64,顾名思义,是基于64种可视字符的编码方式.这64种符号由A ...
- AIM Tech Round (Div. 1) C. Electric Charges 二分
C. Electric Charges 题目连接: http://www.codeforces.com/contest/623/problem/C Description Programmer Sas ...
- NHibernate 之持久化类、拦截器 (第二篇)
一.持久化类中成员标量的要求 作为被NHibernate使用的持久化类,必须满足以下几点要求: 1.声明读写属性 在NHibernate的使用中,持久化类的成员变量必须声明对应的属性,NHiberna ...
- PatchGuard Disabled V3
https://twitter.com/Fyyre http://www.m5home.com/bbs/thread-5893-1-1.html http://pan.baidu.com/share/ ...
- minGW cygwin gnuwin32
首先,三个的官方网站分别是: minGW:http://www.mingw.org cygwin: http://www.cygwin.com gnuwin32: https://sourcefo ...
- Inno Setup入门(十九)——Inno Setup类参考(5)
单选按钮 单选按钮在安装中也很常见,例如同一个程序可以选择安装不同的性质的功能,例如选择32位或者64位等,两者是排他性的,因此可以通过单选按钮(RadioButton)来实现,在同一个容器中放置的单 ...
- iOS开发利器-CocoaPods安装和使用教程
新博客http://www.liuchendi.com 开发iOS项目时肯定会用到许多第三方项目,比如说:ASIHttprequest,JSONKit等等,一些类库可能又关联着其他类库,如果超过一定的 ...