双连通分量 Road Construction POJ - 3352

@[双连通分量]

题意:

有一个 n 个点 m 条边的无向图,问至少添加几条边,能让该图任意缺少一条边后还能相互连通。

双连通分量定义:

在无向连通图中,如果删除该图的任何一个结点都不能改变该图的连通性,则该图为双连通的无向图。一个连通的无向图是双连通的,当且仅当它没有关节点(这里面节点可换成边:分点双连通分量 ,分边双连通分量)。

思路:

首先缩点成树;

与强连通分量缩点有所不同:记录父节点 ,不返回父节点 (意味着一条边只能从任意方向走一次)如果已经走过 ,直接可更新low值(目前理解:若这个点 B 已经走过,出栈后还能再次通过 A 访问到,说明从 B 也能访问到 A ,所以不需要是否在栈中的判断,在强连通分量中,因为是单向,所以只能从 A -> B ,需要是否在栈中的判断)

试了一下加上栈的判断也对:因为访问B的时候直接就通过B 把 A 访问了,不会等到 A 去访问 B 。

缩点成树之后:

统计有 ans 个双连通分量只有一条边且只与一个双连通分量相连,(ans+1)/2 就是至少要加的边数

撸代码

#include<stdio.h>#include<string.h>#include<stack>#include<vector>#include<iostream>#include<algorithm>using namespace std;#define N 1010struct node{    int to,nex;} edge[N*2];int cost[N],dfn[N],low[N],belong[N],head[N];bool instack[N];int in[N];int cnt,cir,index;stack<int>s;vector<int>point[N];void init(){    cnt=0;    cir=0;    index=0;    while(!s.empty())        s.pop();    for(int i=0; i<N; i++)    {        point[i].clear();        head[i]=-1;        in[i]=0;        instack[i]=false;        dfn[i]=0;        low[i]=0;        belong[i]=0;    }}void addEdge(int u,int v){    edge[cnt].to=v;    edge[cnt].nex=head[u];    head[u]=cnt++;}/*求双连通分量*/void Tarjan(int u,int fa){    dfn[u]=low[u]=++index;    instack[u]=true;    s.push(u);    for(int i=head[u]; i!=-1; i=edge[i].nex)    {        int v=edge[i].to;        if(v==fa)            continue;        if(!dfn[v])        {            Tarjan(v,u);            low[u]=min(low[u],low[v]);        }        else //if(instack[v])        {/*走过且不在栈中*/            low[u]=min(low[u],dfn[v]);        }    }    if(low[u]==dfn[u])    {        int node;        ++cir;        do        {            node=s.top();            s.pop();            belong[node]=cir;            point[cir].push_back(node);            instack[node]=false;        }        while(node!=u);    }    return ;}int main(){    int n,m;    while(~scanf("%d%d",&n,&m))    {        init();        int a,b;        for(int i=0; i<m; i++)        {            scanf("%d%d",&a,&b);            addEdge(a,b);            addEdge(b,a);        }        for(int i=1; i<=n; i++)            if(!dfn[i])                Tarjan(i,i);        /*直到每个点所属的强连通分量*///        printf("cir = %d\n",cir);//        for(int i=1;i<=cir;i++)//        {//            printf("cnt [%d]:",i);//            for(int j=0;j<point[i].size();j++)//                printf("%d ",point[i][j]);//            printf("\n");//        }        for(int i=1; i<=n; i++)        {            for(int j=head[i]; j!=-1; j=edge[j].nex)            {                /*!根据统计边 统计连通分量之间的度*/                a=belong[i];                b=belong[edge[j].to];                if(a!=b)                {                    in[a]++;                    in[b]++;                }            }        }        int ans=0;        for(int i=1; i<=cir; i++)        {            if(in[i]==2)                ans++;        }        printf("%d\n",(ans+1)/2);    }    return 0;}

原文地址:https://www.cnblogs.com/HappyKnockOnCode/p/12641167.html

时间: 2024-11-15 17:58:15

双连通分量 Road Construction POJ - 3352的相关文章

Road Construction POJ - 3352 (边双连通分量)

Road Construction POJ - 3352 题意:一个无向图(无重边),问至少还要加多少边使得去掉任意一条边后任意两点仍可互达. 无向图的边双连通分量(无重边) 先用一次dfs标记出割边,然后dfs标记出各联通分量 再根据割边,缩点重新建图,生成一颗树 则答案就是(叶子树+1)/2. 1 #include <iostream> 2 #include <cstdio> 3 #include <algorithm> 4 #include <cstring

POJ 3352 Road Construction POJ 3177 Redundant Paths(边双连通图 Tarjan+缩点)

POJ 3352 Road Construction POJ 3177 Redundant Paths(边双连通图 Tarjan+缩点) ACM 题目地址: POJ 3352 Road Construction POJ 3177 Redundant Paths 题意: 问要添加几条边才能使所给无向图图变成边双连通图. 分析: 边连通度:使无向图G不连通的最少删边数量为其边连通度. 边双连通图:边连通度大于1的无向图. 首先缩点,让图变成一个DAG. 现在问题转化为:在树中至少添加多少条边能使图变

Poj 3352 Road Construction &amp; Poj 3177 Redundant Paths(边双连通分量+缩点)

Road Construction Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 9465   Accepted: 4699 Description It's almost summer time, and that means that it's almost summer construction time! This year, the good people who are in charge of the ro

POJ 3352 Road Construction&amp;&amp; POJ 3177 Redundant Paths 双联通分量

大意:给定n点,和m条边的关系图中的一些边随时可能施工导致不能够通过,所以至少加多少条边才能够使得途中任意两条边联通? 思路:很明显只要图中的任意两点都是两条边来链接即可.那么我们可以先缩点构建新图,然后统计出度为1的点的个数ans,那么需要加的边数就是(ans+1)/2条; (PS;因为建图是双向的图所以,在Tarjan缩点的时候就需要遇到临边便越过,并且判断是不是同一个联通分支用num比较!) #include<map> #include<queue> #include<

POJ 3352 Road Construction(边—双连通分量)

http://poj.org/problem?id=3352 题意: 给出一个图,求最少要加多少条边,能把该图变成边—双连通. 思路:双连通分量是没有桥的,dfs一遍,计算出每个结点的low值,如果相等,说明属于同一个双连通分量. 接下来把连通分量缩点,然后把这些点连边. 对于一棵无向树,我们要使得其变成边双连通图,需要添加的边数 == (树中度数为1的点的个数+1)/2. 1 #include<iostream> 2 #include<algorithm> 3 #include&

POJ - 3352 Road Construction(边双连通分量)

题目大意:给出一张无向图,问添加多少边才能使得这张无向图变成边双连通分量 解题思路:先求出所有的边双连通分量,再将边双连通缩成一个点,通过桥连接起来,这样就形成了一棵无根树了 现在的问题是,将这颗无根树变成边双连通分量 网上的解释是:统计出树中度为1的节点的个数,即为叶节点的个数,记为leaf.则至少在树上添加(leaf+1)/2条边,就能使树达到边二连通,所以至少添加的边数就是(leaf+1)/2.具体方法为,首先把两个最近公共祖先最远的两个叶节点之间连接一条边,这样可以把这两个点到祖先的路径

POJ 3352 Road Construction(边双连通分量,桥,tarjan)

题解转自http://blog.csdn.net/lyy289065406/article/details/6762370   文中部分思路或定义模糊,重写的红色部分为修改过的. 大致题意: 某个企业想把一个热带天堂岛变成旅游胜地,岛上有N个旅游景点,保证任意2个旅游景点之间有路径连通的(可间接连通).而为了给游客提供更方便的服务,该企业要求道路部门在某些道路增加一些设施. 道路部门每次只会选择一条道路施工,在该条道路施工完毕前,其他道路依然可以通行.然而有道路部门正在施工的道路,在施工完毕前是

POJ 3352 Road Construction 使得无向图边变双连通图

点击打开链接 Road Construction Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 8168   Accepted: 4106 Description It's almost summer time, and that means that it's almost summer construction time! This year, the good people who are in charge of

POJ 3177 Redundant Paths POJ 3352 Road Construction(双连通)

POJ 3177 Redundant Paths POJ 3352 Road Construction 题目链接 题意:两题一样的,一份代码能交,给定一个连通无向图,问加几条边能使得图变成一个双连通图 思路:先求双连通,缩点后,计算入度为1的个数,然后(个数 + 1) / 2 就是答案(这题由于是只有一个连通块所以可以这么搞,如果有多个,就不能这样搞了) 代码: #include <cstdio> #include <cstring> #include <algorithm&