hihoCoder #1184 : 连通性二·边的双连通分量(边的双连通分量模板)

Posted Yeader

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了hihoCoder #1184 : 连通性二·边的双连通分量(边的双连通分量模板)相关的知识,希望对你有一定的参考价值。

#1184 : 连通性二·边的双连通分量

时间限制:10000ms
单点时限:1000ms
内存限制:256MB

描述

在基本的网络搭建完成后,学校为了方便管理还需要对所有的服务器进行编组,网络所的老师找到了小Hi和小Ho,希望他俩帮忙。

老师告诉小Hi和小Ho:根据现在网络的情况,我们要将服务器进行分组,对于同一个组的服务器,应当满足:当组内任意一个连接断开之后,不会影响组内服务器的连通性。在满足以上条件下,每个组内的服务器数量越多越好。

比如下面这个例子,一共有6个服务器和7条连接:

技术分享图片

其中包含2个组,分别为{1,2,3},{4,5,6}。对{1,2,3}而言,当1-2断开后,仍然有1-3-2可以连接1和2;当2-3断开后,仍然有2-1-3可以连接2和3;当1-3断开后,仍然有1-2-3可以连接1和3。{4,5,6}这组也是一样。

技术分享图片

老师把整个网络的情况告诉了小Hi和小Ho,小Hi和小Ho要计算出每一台服务器的分组信息。

   

提示:边的双连通分量

 

输入

第1行:2个正整数,N,M。表示点的数量N,边的数量M。1≤N≤20,000, 1≤M≤100,000

第2..M+1行:2个正整数,u,v。表示存在一条边(u,v),连接了u,v两台服务器。1≤u<v≤N

保证输入所有点之间至少有一条连通路径。

输出

第1行:1个整数,表示该网络的服务器组数。

第2行:N个整数,第i个数表示第i个服务器所属组内,编号最小的服务器的编号。比如分为{1,2,3},{4,5,6},则输出{1,1,1,4,4,4};若分为{1,4,5},{2,3,6}则输出{1,2,2,1,1,2}

代码

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #include<vector>
 5 #include<stack>
 6 #include<algorithm>
 7 using namespace std;
 8 const int N=2e4+5;
 9 vector<int>v[N];
10 stack<int>sk;
11 int num,cnt;
12 int low[N],dfn[N],fa[N],ans[N];
13 
14 void init(){
15     cnt=num=0;
16     memset(dfn,0,sizeof(dfn));
17     memset(low,0,sizeof(low));
18     memset(fa,0,sizeof(fa));
19     memset(ans,0,sizeof(ans));
20 }
21 
22 void dfs(int u,int f){
23     low[u]=dfn[u]=++cnt;
24     sk.push(u);
25     for(int i=0;i<v[u].size();i++){
26         int t=v[u][i];
27         if(!dfn[t]){
28             dfs(t,u);
29             low[u]=min(low[u],low[t]);
30         }
31         else if(t!=f) low[u]=min(low[u],dfn[t]);      //无向图不需要判断是否在栈中
32     }
33     // 因为low[u] == dfn[u],对(parent[u],u)来说有dfn[u] > dfn[ parent[u] ],因此low[u] > dfn[ parent[u] ]
34     // 所以(parent[u],u)一定是一个桥,那么此时栈内在u之前入栈的点和u被该桥分割开
35     // 则u和之后入栈的节点属于同一个组
36     //最后剩下的一个(或者说第一个)组虽然前面没有割边,但是也适用于这个判断
37     if(low[u]==dfn[u]){
38         ++num;
39         while(!sk.empty()){
40             int t=sk.top();
41             sk.pop();
42             fa[t]=num;
43             if(ans[num]==0||ans[num]>t)
44                 ans[num]=t;
45             if(t==u)
46                 break;
47         }
48     }
49 }
50 
51 int main(){
52     int n,m;
53     while(~scanf("%d%d",&n,&m)){
54         init();
55         for(int i=1;i<=m;i++){
56             int a,b;
57             scanf("%d%d",&a,&b);
58             v[a].push_back(b);
59             v[b].push_back(a);
60         }
61         dfs(1,-1);
62         printf("%d\n",num);
63         for(int i=1;i<=n;i++){
64             printf("%d%c",ans[fa[i]],i==n?\n: );
65         }
66     }
67     return 0;
68 }

 


以上是关于hihoCoder #1184 : 连通性二·边的双连通分量(边的双连通分量模板)的主要内容,如果未能解决你的问题,请参考以下文章

hihocoder欧拉路径 并查集判连通

hihocoder #1190 : 连通性·四 点双联通分量

hihocoder #1185 : 连通性·三 tarjan

hiho一下第133周 2-SAT·hihoCoder音乐节(2-SAT)(强连通)

HihoCoder 1183 : 连通性一·割边与割点

HihoCoder 1185 : 连通性·三(强连通缩点)