https://www.luogu.org/problem/show?pid=3225

题目描述

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

请写一个程序,用来计算至少需要设置几个救援出口,以及不同最少救援出口的设置方案总数。

输入输出格式

输入格式:

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

输出格式:

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

输入输出样例

输入样例#1:

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
输出样例#1:

Case 1: 2 4
Case 2: 4 1

说明

Case 1 的四组解分别是(2,4),(3,4),(4,5),(4,6);

Case 2 的一组解为(4,5,6,7)。

很明显需要求出割点。屏蔽割点后、

①若一个双联通中没有割点,那么最少需要两个出口(炸了一个去另一个)

②若一个双联通能连到一个割点,那么最少需要在双联通中设置一个出口,炸了割点还能逃生、

③若一个双联通能连到多余一个割点,那么不管怎么炸,都能从没炸的割点跑到别的分量里

 #include <cstring>
#include <cstdio> const int N();
int head[N],sumedge=;
struct Edge
{
int v,next;
Edge(int v=,int next=):v(v),next(next){}
}edge[N*N/];
inline void ins(int u,int v)
{
edge[++sumedge]=Edge(v,head[u]);
head[u]=sumedge;
edge[++sumedge]=Edge(u,head[v]);
head[v]=sumedge;
} #define max(a,b) (a>b?a:b)
#define min(a,b) (a<b?a:b)
int tim,dfn[N],low[N];
int cutpoint[N];
void Tarjan(int u,int pre)
{
dfn[u]=low[u]=++tim;
int sumtredge=,if_point=;
for(int v,i=head[u];i;i=edge[i].next)
{
v=edge[i].v;
if((i^)==pre) continue;
if(!dfn[v])
{
sumtredge++;
Tarjan(v,i);
if(low[v]>=dfn[u]) if_point=;
low[u]=min(low[u],low[v]);
}
else low[u]=min(low[u],dfn[v]);
}
if(!pre)
{
if(sumtredge>) cutpoint[u]=;
}
else if(if_point) cutpoint[u]=;
} int sumcol,point,cnt,vis[N],col[N];
void DFS(int u)
{
point++;
vis[u]=;
col[u]=sumcol;
for(int v,i=head[u];i;i=edge[i].next)
{
v=edge[i].v;
if(cutpoint[v]&&col[v]!=sumcol) cnt++,col[v]=sumcol;
if(!vis[v]&&!cutpoint[v]) DFS(v);
}
} inline void init()
{
tim=sumcol=; sumedge=;
memset(dfn,,sizeof(dfn));
memset(col,,sizeof(col));
memset(vis,,sizeof(vis));
memset(low,,sizeof(low));
memset(head,,sizeof(head));
memset(edge,,sizeof(edge));
memset(cutpoint,,sizeof(cutpoint));
} int main()
{
for(int m,n=,k=;scanf("%d",&m)&&m;k++,n=,init())
{
for(int u,v;m--;ins(u,v))
scanf("%d%d",&u,&v),n=max(n,max(u,v));
for(int i=;i<=n;i++)
if(!dfn[i]) Tarjan(i,);
long long ans_sum=;
int ans_cnt=;
for(int i=;i<=n;i++)
{
if(vis[i]||cutpoint[i]) continue;
point=cnt=;
sumcol++,DFS(i);
if(!cnt) ans_cnt+=,ans_sum*=(long long)(point*(point-)>>);
if(cnt==) ans_cnt++,ans_sum*=(long long)point;
}
printf("Case %d: %d %lld\n",k,ans_cnt,ans_sum);
}
return ;
}

最新文章

  1. linux配置java开发环境
  2. hadoop中master免登录slave
  3. Android IOS WebRTC 音视频开发总结(五四)-- WebRTC标准之父谈WebRTC
  4. 算法:C++排列组合
  5. bhrs报表年结步骤
  6. DIV当textarea使用,在聚焦的时候将光标移动到内容的末尾
  7. .NET中常见对象
  8. Java中的static关键字
  9. Android -- NestedScrolling滑动机制
  10. PAT1094:The Largest Generation
  11. Keras的一些功能函数
  12. Xamarin.Forms 开发资源集合
  13. HATEOAS约束
  14. 简述iproute家族命令
  15. Python实现将爱词霸每日一句定时推送至微信
  16. SpringBoot整合Druid(阿里巴巴)数据源
  17. 第一次java程序测试感受
  18. Scrapy结构
  19. 2017.5.10 MapReduce内部逻辑
  20. 基于windows的mongodb不支持mongodbsniff等其他一些功能

热门文章

  1. Chisel Tutorial(一)——Chisel介绍
  2. AnyForWeb告诉你什么才是“最好的”编程语言
  3. hdu5386 Cover
  4. [IOS]mac以太网连接
  5. Unity特殊目录和脚本编译顺序
  6. iOS-UIImageView载入网络下载的图片(异步+多线程)
  7. Ubuntu桌面基础介绍
  8. 剑指offer——03从尾至头打印列表(Python3)
  9. linux 下Redis 5.0主从复制(一主二从)哨兵模式的搭建
  10. category的概念