2730: [HNOI2012]矿场搭建

Time Limit: 10 Sec  Memory Limit: 128 MB
Submit: 1602  Solved: 751
[Submit][Status][Discuss]

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

Source

day1

Solution

对于删除一个点,其余点要有出路,显然是和割点有关,那么我们求出所有割点,以及双连通分量。

对于一个双联通分量中,如果我们只删除一个割点或非割点,那么如果还有其余割点能使其余点到关键点,那么显然是不需要额外考虑的;如果没有其余的割点令我们到关键点,那么我们需要新建额外的关键点

分情况讨论,如果一个双联通分量里,有两个及以上割点,那么这个双联通分量里面是不需要额外建的

如果一个双联通分量里只有一个或没有割点,那么我们只需要再建一个即可。

至于方案数,可以利用乘法原理,我们把每个连通分量里的每个可以用来建成关键点的点用乘法统计起来即可

Code

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<vector>
using namespace std;
#define LL long long
inline int read()
{
int x=,f=; char ch=getchar();
while (ch<'' || ch>'') {if (ch=='-') f=-; ch=getchar();}
while (ch>='' && ch<='') {x=x*+ch-''; ch=getchar();}
return x*f;
}
#define MAXM 1010
#define MAXN 80010
int N,M,tot;
LL sum;
struct EdgeNode{int next,to,from;}edge[MAXM<<];
int head[MAXN],cnt=;
void AddEdge(int u,int v) {cnt++; edge[cnt].next=head[u]; head[u]=cnt; edge[cnt].to=v;}
void InsertEdge(int u,int v) {AddEdge(u,v); AddEdge(v,u);}
#define Pa pair<int,int>
vector<int>BCC[MAXN];
Pa st[MAXM]; int top;
int dfn[MAXN],low[MAXN],dfsn,cut[MAXN],bcc,belong[MAXN];
void Tarjan(int now,int last)
{
dfn[now]=low[now]=++dfsn; int son=;
for (int i=head[now]; i; i=edge[i].next)
if (!dfn[edge[i].to])
{
st[++top]=make_pair(now,edge[i].to); son++;
Tarjan(edge[i].to,now); low[now]=min(low[now],low[edge[i].to]);
if (dfn[now]<=low[edge[i].to])
{
cut[now]=; bcc++; BCC[bcc].clear(); int tnow=-,tto=-;
while ()
{
tnow=st[top].first,tto=st[top].second; top--;
if (belong[tnow]!=bcc) BCC[bcc].push_back(tnow),belong[tnow]=bcc;
if (belong[tto]!=bcc) BCC[bcc].push_back(tto),belong[tto]=bcc;
if (tnow==now && tto==edge[i].to) break;
}
}
}
else if (dfn[edge[i].to]<dfn[now] && edge[i].to!=last)
st[++top]=make_pair(now,edge[i].to),low[now]=min(low[now],dfn[edge[i].to]);
if (last< && son==) cut[now]=;
}
int main()
{
int cas=;
while (scanf("%d",&M))
{
if (M==) break;
N=; cnt=; memset(head,,sizeof(head)); top=; bcc=; memset(st,,sizeof(st));
memset(dfn,,sizeof(dfn)); memset(low,,sizeof(low)); bcc=,dfsn=;
memset(cut,,sizeof(cut)); memset(belong,,sizeof(belong)); for (int x,y,i=; i<=M; i++) x=read(),y=read(),InsertEdge(x,y),N=max(N,max(x,y));
for (int i=; i<=N; i++) if (!dfn[i]) Tarjan(i,-);
// for (int i=1; i<=N; i++) printf("%d %d %d\n",dfn[i],belong[i],cut[i]);
tot=,sum=;
// printf("%d\n",bcc);
// for (int i=1; i<=bcc; i++) printf("%d ",BCC[i].size()); puts("");
for (int i=,num=,sz=BCC[i].size()-; i<=bcc; i++,num=,sz=BCC[i].size()-)
{
// printf("%d %d %I64d\n",num,tot,sum);
for (int j=; j<=sz; j++) if (cut[BCC[i][j]]) num++;
if (num==) tot++,sum*=(LL)(BCC[i].size()-num);
if (bcc==) {tot=; sum=(LL)BCC[].size()*(BCC[].size()-)/; break;}
}
printf("Case %d: %d %lld\n",++cas,tot,sum);
}
return ;
}

最新文章

  1. LevelDB库简介
  2. 实现跨域的N种方法
  3. Yii查看执行的SQL
  4. 使用xml方式定义补间动画
  5. MFC图形图像
  6. mysql中相关,无关子查询,表与表之间的关系以及编码和乱码的解决
  7. Android Studio-ApplicationId 与 PackageName
  8. NFS服务器端配置
  9. 洛谷 [P2024] 食物链
  10. 服务消费者Feign和Ribbon的区别
  11. Yslow web性能测试插件
  12. Choosing web framework: ASP.NET MVC vs Django Python vs Ruby on Rails(转载)
  13. 利用win10自带的虚拟机Hyper-V安装Centos7的步骤教程
  14. JavaScript笔记 #08# 用Regex辅助生成文章目录 V2.0
  15. GBDT原理
  16. Python 并发编程(管道,事件,信号量,进程池)
  17. VMware 克隆多台Linux机器并配置IP
  18. 在HTML标签的事件后面直接写JS代码
  19. Hessain 方法重载
  20. bzoj1600 [Usaco2008 Oct]建造栅栏(递推)

热门文章

  1. 关于OC中的小数精确计算---NSDecimalNumber
  2. 从零自学Hadoop(12):Hadoop命令中
  3. 关于Redis数据过期策略
  4. PL/SQL入门理解(一)
  5. Oracle数据库穿越防火墙访问
  6. 浅谈Java中的对象和引用
  7. 解决www.github.com访问太慢的问题
  8. winform修改、打开窗体、构造函数传值
  9. linux ubuntu14 更改为root用户登录
  10. VS一直停留在“正在还原nuget程序包”