Description

煤矿工地可以看成是由隧道连接挖煤点组成的无向图。为安全起见,希望在工地发生事故时所有挖煤点的工人都能有一条出路逃到救援出口处。于是矿主决定在某些挖煤点设立救援出口,使得无论哪一个挖煤点坍塌之后,其他挖煤点的工人都有一条道路通向救援出口。请写一个程序,用来计算至少需要设置几个救援出口,以及不同最少救援出口的设置方案总数。

Input

输入文件有若干组数据,每组数据的第一行是一个正整数 N(N≤500),表示工地的隧道数,接下来的 N 行每行是用空格隔开的两个整数 S 和 T,表示挖       S 与挖煤点 T 由隧道直接连接。输入数据以 0 结尾。

Output

输入文件中有多少组数据,输出文件 output.txt 中就有多少行。每行对应一组输入数据的 结果。其中第 i 行以 Case i: 开始(注意大小写,Case 与 i 之间有空格,i 与:之间无空格,: 之后有空格),其后是用空格隔开的两个正整数,第一个正整数表示对于第 i 组输入数据至少需 要设置几个救援出口,第二个正整数表示对于第 i 组输入数据不同最少救援出口的设置方案总 数。输入数据保证答案小于 2^64。输出格式参照以下输入输出样例。

Sample Input

9
1 3
4 1
3 5
1 2
2 6
1 5
6 3
1 6
3 2
6
1 2
1 3
2 4
2 5
3 6
3 7
0

Sample Output

Case 1: 2 4
Case 2: 4 1

HINT

Case 1 的四组解分别是(2,4),(3,4),(4,5),(4,6);
Case 2 的一组解为(4,5,6,7)。
———————————————————————————————
这道题我们发现对于每一个点双联通分量 如果他没有割点 那么他就是一个单独的块 
对于这样的块 我们应该给他设两个出口 防止一个崩了 而对于只有一个联通块的 我们应该设一个不再割点的出口
这样如果出口崩掉了还可以去别的块 而对于有大于一个割点 不用设出口 他可以随便去别的块
然后乘法原理就可以辣 (吐槽:点双写起来贼丑
  1. #include<cstdio>
  2. #include<cstring>
  3. #include<algorithm>
  4. #define LL long long
  5. using std::max;
  6. using std::min;
  7. const int M=1e3+;
  8. int read(){
  9. int ans=,f=,c=getchar();
  10. while(c<''||c>''){if(c=='-') f=-; c=getchar();}
  11. while(c>=''&&c<=''){ans=ans*+(c-''); c=getchar();}
  12. return ans*f;
  13. }
  14. int n,m;
  15. int first[M],cnt,buck[M];
  16. struct node{int from,to,next;}e[*M];
  17. void ins(int a,int b){e[++cnt]=(node){a,b,first[a]}; first[a]=cnt;}
  18. void insert(int a,int b){ins(a,b); ins(b,a);}
  19. int dfn[M],low[M],T,iscut[M];
  20. int hc,sz[M],color[M],stk[M],top;
  21. void clear(){
  22. n=; cnt=; T=; hc=; top=;
  23. memset(iscut,,sizeof(iscut));
  24. memset(first,,sizeof(first));
  25. memset(dfn,,sizeof(dfn));
  26. memset(low,,sizeof(low));
  27. memset(sz,,sizeof(sz));
  28. memset(color,,sizeof(color));
  29. memset(buck,,sizeof(buck));
  30. }
  31. void tarjan(int x,int fa){
  32. low[x]=dfn[x]=++T;
  33. int child=;
  34. for(int i=first[x];i;i=e[i].next){
  35. int now=e[i].to;
  36. if(!dfn[now]){
  37. stk[++top]=i;
  38. child++;
  39. tarjan(now,x);
  40. low[x]=min(low[x],low[now]);
  41. if(low[now]>=dfn[x]){
  42. iscut[x]=;
  43. hc++;
  44. while(){
  45. int k=stk[top--];
  46. if(color[e[k].from]!=hc){
  47. if(color[e[k].from]) buck[color[e[k].from]]++;
  48. sz[hc]++; color[e[k].from]=hc;
  49. }
  50. if(color[e[k].to]!=hc){
  51. if(color[e[k].to]) buck[color[e[k].to]]++;
  52. sz[hc]++; color[e[k].to]=hc;
  53. }
  54. if(e[k].from==x&&e[k].to==now) break;
  55. }
  56. }
  57. }
  58. else if(dfn[now]<dfn[x]&&now!=fa) stk[++top]=i,low[x]=min(low[x],dfn[now]);
  59. }
  60. if(fa==-&&child==) iscut[x]=;
  61. }
  62. int main(){
  63. int x,y,h=;
  64. while(scanf("%d",&m)==&&m){
  65. ++h; clear();
  66. for(int i=;i<=m;i++){
  67. x=read(); y=read();
  68. insert(x,y); n=max(n,max(x,y));
  69. }
  70. for(int i=;i<=n;i++)if(!dfn[i]) tarjan(i,-);
  71. for(int i=;i<=n;i++)if(iscut[i]) buck[color[i]]++;
  72. LL ans=;
  73. int tot=;
  74. for(int i=;i<=hc;i++){
  75. if(buck[i]==) ans*=max(sz[i]-,),tot++;
  76. if(!buck[i]) ans*=sz[i]*(sz[i]-)/,tot+=;
  77. }
  78. printf("Case %d: %d %lld\n",h,tot,ans);
  79. }
  80. return ;
  81. }

bzoj 2730: [HNOI2012]矿场搭建——tarjan求点双的更多相关文章

  1. BZOJ 2730: [HNOI2012]矿场搭建( tarjan )

    先tarjan求出割点.. 割点把图分成了几个双连通分量..只需dfs找出即可. 然后一个bcc有>2个割点, 那么这个bcc就不用建了, 因为一定可以走到其他救援出口. 只有一个割点的bcc就 ...

  2. 【刷题】BZOJ 2730 [HNOI2012]矿场搭建

    Description 煤矿工地可以看成是由隧道连接挖煤点组成的无向图.为安全起见,希望在工地发生事故时所有挖煤点的工人都能有一条出路逃到救援出口处.于是矿主决定在某些挖煤点设立救援出口,使得无论哪一 ...

  3. bzoj 2730: [HNOI2012]矿场搭建【tarjan】

    先tarjan找割点和点双连通分量,然后对一个点双,如果没有割点,那么需要建立两个出口(割掉一个另一个备用):如果只有一个割点,出口可以设立在任意一个非割点的地方:如果有两个及以上个割点,就不用建出口 ...

  4. bzoj 2730: [HNOI2012]矿场搭建

    #include<cstdio> #include<cstring> #include<iostream> #define M 508 using namespac ...

  5. 【BZOJ】2730: [HNOI2012]矿场搭建【Tarjan找割点】【分联通块割点个数】

    2730: [HNOI2012]矿场搭建 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 3230  Solved: 1540[Submit][Stat ...

  6. 【BZOJ2730】[HNOI2012]矿场搭建 Tarjan

    [BZOJ2730][HNOI2012]矿场搭建 Description 煤矿工地可以看成是由隧道连接挖煤点组成的无向图.为安全起见,希望在工地发生事故时所有挖煤点的工人都能有一条出路逃到救援出口处. ...

  7. BZOJ 2730 矿场搭建 Tarjan求割点

    思路: Tarjan求出来点双&割点 判一判就行了 //By SiriusRen #include <stack> #include <cstdio> #include ...

  8. BZOJ2730 [HNOI2012]矿场搭建 - Tarjan割点

    Solution 输入中没有出现过的矿场点是不用考虑的, 所以不用考虑只有 一个点 的点双联通分量. 要使某个挖矿点倒塌, 相当于割去这个点, 所以我们求一遍割点和点双联通分量. 之后的点双联通分量构 ...

  9. [BZOJ2730][HNOI2012]矿场搭建(求割点)

    题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=2730 分析: 如果坍塌的点不是割点,那没什么影响,主要考虑坍塌的点是割点的情况. 显然 ...

随机推荐

  1. Memcache+Cookie解决分布式系统共享登录状态

    Memcached高性能的,分布式的内存对象缓存系统,用于在动态应用中减少数据库负载,提升访问速度.Memcached能够用来存储各种格式的数据,包括图像.视频.文件以及数据库检索的结果等. Memc ...

  2. Java中I/O流之数据流

    Java 中的数据流: 对于某问题:将一个 long 类型的数据写到文件中,有办法吗?    转字符串 → 通过 getbytes() 写进去,费劲,而且在此过程中 long 类型的数需要不断地转换. ...

  3. 原生javascript自定义input[type=radio]效果

    2018年6月27日 更新 找到最为简单的仅仅使用css3的方案 <!DOCTYPE html> <html lang="en"> <head> ...

  4. BZOJ 2120 数颜色(树状数组套主席树)

    1A啊,激动. 首先,不修改的情况下可以直接用主席树搞,修改的话,直接用主席树搞一次修改的情况下复杂度是O(nlogn)的. 就像你要求区间和一样,用前缀和查询是O(1),修改是O(n),只不过主席树 ...

  5. 【题解】Atcoder AGC#03 E-Sequential operations on Sequence

    仙题膜拜系列...首先我们可以发现:如果在截取了一段大的区间之后再截取一段小的区间,显然是没有什么用的.所以我们可以将操作序列变成单调递增的序列. 然后怎么考虑呢?启示:不一定要考虑每一个数字出现的次 ...

  6. 【题解】Atcoder ARC#97 F-Monochrome Cat

    好zz啊我……(:д:) 首先我们可以删掉所有只有黑色节点的子树(一定不会遍历到), 且注意到每一个点你一定只会经过一遍.然后又考虑如果起点和终点相同,那么总次数实际上已经固定了:就是遍历整棵树(每一 ...

  7. ssm框架pom.xml

    <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/20 ...

  8. BZOJ2223:[Coci2009]PATULJCI——题解

    http://www.lydsy.com/JudgeOnline/problem.php?id=2223 Description Sample Input 10 3 1 2 1 2 1 2 3 2 3 ...

  9. 20165218 实验一 Java开发环境的熟悉

    实验一 Java开发环境的熟悉 课程:java程序设计 姓名:赵冰雨 学号:20165218 指导教师:娄嘉鹏 实验日期:2018.4.2 实验密级:Java开发环境的熟悉 实验内容.步骤与体会: ( ...

  10. 【神仙题】【P4885】 灭顶之灾

    传送门 Description 请将题目名称的首字母连起来读 Scarlet有一张$n*m$的神秘表格.现在Scarlet向表格中填数字,她会从第一行中的某个格子起,按照从左往右,从上往下的顺序依次填 ...