zoukankan      html  css  js  c++  java
  • 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
    ",num);
    63         for(int i=1;i<=n;i++){
    64             printf("%d%c",ans[fa[i]],i==n?'
    ':' ');
    65         }
    66     }
    67     return 0;
    68 }
    
    
  • 相关阅读:
    C++ Opencv 傅里叶变换的代码实现及关键函数详解
    C++ Opencv createTrackbar()创建滑动条实现对比度、亮度调节及注意事项
    C++ Opencv Mat类型使用的几个注意事项及自写函数实现Laplace图像锐化
    C++ Opencv split()通道分离函数 merge()通道合并函数 使用操作详解
    如何将nupkg文件安装到VS2017
    [C#]如何访问及调用类中私有成员及方法
    [C#]使用Redis来存储键值对(Key-Value Pair)
    [C#]使用log4net记录日志
    [C#]使用Quartz.NET来创建定时工作任务
    [C#]使用IFormattable接口来实现字符串格式化
  • 原文地址:https://www.cnblogs.com/fu3638/p/8635542.html
Copyright © 2011-2022 走看看