BZOJ_2730_ [HNOI2012]矿场搭建_点双联通分量

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>
#include <cstring>
#include <algorithm>
#include <vector>
using namespace std;
#define N 100050
typedef long long ll;
vector<int>V[N];
int head[N],to[N<<1],nxt[N<<1],cnt=1;
int vis[N<<1],S[N<<1],top,bl[N<<1],bcc,dfn[N],low[N],rt,tot,iscut[N],fa[N],n,m;
inline void add(int u,int v) {
to[++cnt]=v; nxt[cnt]=head[u]; head[u]=cnt;
}
void dfs(int x) {
int i,num=0;
dfn[x]=low[x]=++tot;
for(i=head[x];i;i=nxt[i]) {
if(vis[i]) continue;
vis[i]=vis[i^1]=1;
num++;
S[++top]=i;
if(!dfn[to[i]]) {
dfs(to[i]);
low[x]=min(low[x],low[to[i]]);
if(dfn[x]<=low[to[i]]) {
iscut[x]=1;
int t;
++bcc; V[bcc].clear();
do {
t=S[top--];
int u=to[t^1],v=to[t];
if(fa[u]!=bcc) fa[u]=bcc,V[bcc].push_back(u);
if(fa[v]!=bcc) fa[v]=bcc,V[bcc].push_back(v);
}while(t!=i);
}
}else {
low[x]=min(low[x],dfn[to[i]]);
}
}
if(rt==x&&num<2) iscut[x]=0;
}
int Cas;
void solve() {
n=0;
Cas++;
memset(head,0,sizeof(head)); cnt=1;
memset(iscut,0,sizeof(iscut));
memset(dfn,0,sizeof(dfn));
memset(fa,0,sizeof(fa));
memset(vis,0,sizeof(vis));
top=0; bcc=0;
int i,x,y,j;
for(i=1;i<=m;i++) scanf("%d%d",&x,&y),add(x,y),add(y,x),n=max(n,max(x,y));
for(i=1;i<=n;i++) {
if(!dfn[i]) {
dfs(rt=i);
}
}
int ans1=0; ll ans2=1;
// for(i=1;i<=n;i++) printf("%d\n",iscut[i]);
// printf("%d\n",bcc);
for(i=1;i<=bcc;i++) {
int lim=V[i].size(),num=0;
for(j=0;j<lim;j++) {
// printf("%d\n",V[i][j]);
if(iscut[V[i][j]]) num++;
}
// puts("FUCK");
if(num==1) ans1++,ans2=ans2*(lim-1);
}
if(bcc==1) ans1=2,ans2=1ll*(V[1].size()-1)*(V[1].size())/2;
printf("Case %d: %d %lld\n",Cas,ans1,ans2);
}
int main() {
while(scanf("%d",&m)!=EOF&&m) solve();
}

最新文章

  1. JAVA中List 排序
  2. 利用office2000组件进行填充打印报不支持集合。 (Exception from HRESULT: 0x80020011 (DISP_E_NOTACOLLECTION))
  3. Qt搭建多线程Server
  4. |原创|unity 4.3 2D功能SpriteRenderer修改颜色的方法
  5. oop典型应用:实体类
  6. C#泛型-模板特化
  7. Maven 建立web项目 The import javax.servlet cannot be resolved
  8. open source e-business software - prestashop
  9. Android关于实现EditText中加多行下划线的的一种方法
  10. RabbitMQ基础总结
  11. bzoj1090:[SCOI2003]字符串折叠
  12. 自定义构造方法和description方法
  13. Mac 上开启一个简单的服务器
  14. Angular4.0.0正式版发布
  15. Python基础学习参考(七):字典和集合
  16. Codeforces Round #483 (Div. 2) D. XOR-pyramid
  17. .Net Core 中间件之主机地址过滤(HostFiltering)源码解析
  18. windows server 2012 流媒体服务器搭建(直播与点播)
  19. python locust 性能测试:locust 参数化(list) ---循环取数据,数据可重复使用
  20. 关于解决JetBrains 激活问题

热门文章

  1. [转] RabbitMQ介绍
  2. PS常用平面设计制作尺寸
  3. android客户端向服务器端验证登陆方法的实现2
  4. javascript变量初始化位置
  5. Android 大众点评的接入
  6. centos创建本地yum仓库
  7. SQLMAP源码分析(一)
  8. Struts2拦截器 解决登录问题
  9. 目标检测之hough forest---霍夫森林(Hough Forest)目标检测算法
  10. 多媒体开发之--- Live555 server 获取不到本地ip 全为0