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)。

Solution

先对题目中给出的图跑一遍Tarjan,把点双和割点都求出来

然后我们发现对于每个点双,存在三种情况

1)点双中不存在割点,那么我们就要在其中最少建2个,这样就算其中有一个塌了,由于这是个点双,其它所有点还是与救生点是连通的

2)点双中只存在一个割点,那么就在除割点之外任意一个点建1个,这样如果割点塌了,其它点可以赶到救援出口;如果是救援出口塌了,那么其它点可以通过割点走到其它联通块去

3)点双中存在一个以上的割点,那么无论是那个点塌了,其它点都可以通过(剩下的)割点走到其它联通块去

对于方案数的话,用组合数可以直接算

#include<bits/stdc++.h>
using namespace std;
#define ll long long
const int MAXN=+;
int n,m,e,to[MAXN<<],nex[MAXN<<],beg[MAXN],out[MAXN],DFN[MAXN],LOW[MAXN],Visit_Num,Be[MAXN],cnt,cut[MAXN],cas,val[MAXN],C[MAXN][MAXN];
ll ans1,ans2;
stack<int> s;
vector<int> point[MAXN];
inline void read(int &x)
{
int data=,w=;
char ch=;
while(ch!='-'&&(ch<''||ch>''))ch=getchar();
if(ch=='-')w=-,ch=getchar();
while(ch>=''&&ch<='')data=(data<<)+(data<<)+(ch^''),ch=getchar();
x=data*w;
}
inline void preinit()
{
C[][]=;
for(register int i=;i<MAXN;++i)
{
C[i][]=;
for(register int j=;j<=i;++j)C[i][j]=C[i-][j]+C[i-][j-];
}
}
inline void init()
{
n=;ans1=;ans2=;cnt=;e=;
memset(Be,,sizeof(Be));
memset(beg,,sizeof(beg));
memset(cut,,sizeof(cut));
memset(DFN,,sizeof(DFN));
memset(LOW,,sizeof(LOW));
memset(val,,sizeof(val));
while(!s.empty())s.pop();
}
inline void insert(int x,int y)
{
to[++e]=y;
out[e]=x;
nex[e]=beg[x];
beg[x]=e;
}
inline void Tarjan(int x,int f)
{
int ch=;
DFN[x]=LOW[x]=++Visit_Num;
for(register int i=beg[x];i;i=nex[i])
if(to[i]==f)continue;
else if(!DFN[to[i]])
{
s.push(i);
ch++;
Tarjan(to[i],x);
LOW[x]=min(LOW[x],LOW[to[i]]);
if(LOW[to[i]]>=DFN[x])
{
cut[x]=;
int temp;
++cnt;
point[cnt].clear();
do{
temp=s.top();
s.pop();
if(Be[out[temp]]!=cnt)
{
Be[out[temp]]=cnt;
point[cnt].push_back(out[temp]);
}
if(Be[to[temp]]!=cnt)
{
Be[to[temp]]=cnt;
point[cnt].push_back(to[temp]);
}
}while(out[temp]!=x||to[temp]!=to[i]);
}
}
else if(DFN[to[i]]<DFN[x])
{
s.push(i);
LOW[x]=min(LOW[x],DFN[to[i]]);
}
if(!f&&ch<)cut[x]=;
}
inline void statistic()
{
for(register int i=;i<=cnt;++i)
{
for(register int j=;j<point[i].size();++j)
if(cut[point[i][j]])val[i]++;
if(val[i]==)ans1+=2ll,ans2*=(ll)C[point[i].size()][];
else if(val[i]==)ans1++,ans2*=(ll)C[point[i].size()-][];
}
}
int main()
{
preinit();
while(scanf("%d",&m)!=EOF&&m)
{
printf("Case %d:",++cas);
init();
for(register int i=;i<=m;++i)
{
int u,v;
read(u);read(v);
insert(u,v);
insert(v,u);
n=max(n,max(u,v));
}
for(register int i=;i<=n;++i)
if(!DFN[i])Tarjan(i,);
statistic();
printf(" %lld %lld\n",ans1,ans2);
}
return ;
}

2730 矿场搭建

【刷题】BZOJ 2730 [HNOI2012]矿场搭建的更多相关文章

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

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

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

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

  3. bzoj 2730: [HNOI2012]矿场搭建——tarjan求点双

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

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

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

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

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

  6. Tarjan水题系列(2):HNOI2012 矿场搭建

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

  7. bzoj2730 [HNOI2012]矿场搭建 (UVAlive5135 Mining Your Own Business)

    2730: [HNOI2012]矿场搭建 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1147  Solved: 528[Submit][Statu ...

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

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

  9. BZOJ 2730:[HNOI2012]矿场搭建(割点+连通块)

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

随机推荐

  1. zabbix4.0LTS安装配置

    一.环境准备 1.最小化安装centos7.5系统 2.安装常用软件包 yum install -y net-tools vim bash-completion 3.关闭selinux和修改主机名 v ...

  2. OpenGL学习笔记(4) GLM库的使用

    OpenGL和DirextX不一样,没有内置的数学库,于是我们需要找一个第三方库,按照LearnOpenGL的教程我们使用GLM库,可以到他们的官网下载 glm常用的数据类型 vec2 二维向量 ve ...

  3. Mysql数据库的隔离级别

    Mysql数据库的隔离级别有四种 1.read umcommitted   读未提交(当前事务可以读取其他事务没提交的数据,会读取到脏数据) 2.read committed 读已提交(当前事务不能读 ...

  4. [linux] LVM磁盘管理(针对xfs和ext4不同文件系统)

    简单来说就是:PV:是物理的磁盘分区VG:LVM中的物理的磁盘分区,也就是PV,必须加入VG,可以将VG理解为一个仓库或者是几个大的硬盘LV:也就是从VG中划分的逻辑分区如下图所示PV.VG.LV三者 ...

  5. JVM调优(2)

    堆大小设置JVM 中最大堆大小有三方面限制:相关操作系统的数据模型(32-bt还是64-bit)限制:系统的可用虚拟内存限制:系统的可用物理内存限制.32位系统下,一般限制在1.5G~2G:64为操作 ...

  6. 数据库之python操作mysql

    目录 一.pymysql 二.SQLAchemy 2.操作使用 (1)连接数据库 (2)执行原生SQL语句 (3)ORM操作-数据表操作 (4)ORM操作-数据行操作 (5)更多例子 一.pymysq ...

  7. Nginx是如何配置为 Web 服务器的【转载】

    详解 Nginx是如何配置为 Web 服务器的 林涛 发表于:2016-11-29 23:23 分类:WebServer 标签:Nginx,web,web服务器 521次 抽象来说,将 Nginx 配 ...

  8. 第十三次作业psp

    psp 进度条 代码累积折线图 博文累积折线图 psp饼状图

  9. (第十周)Beta Review会议

    项目名:食物链教学工具 组名:奋斗吧兄弟 组长:黄兴 组员:李俞寰.杜桥.栾骄阳.王东涵 Beta Review会议 时间:2016.11.14   10:00——11:30.13:30——15:00 ...

  10. Daily scrum 2015.10.19

    这周是我们团队项目开始的第一周.我们的团队项目是“北航社团平台”,一个致力于打造北航社团资讯整合.社团工作服务与社团商品销售的一站式网络平台. 一.会议内容 1. 总体分工,江昊同学担任项目PM,王若 ...