hszxoj 矿场搭建

  • 题目描述

    原题来自:HNOI 2012

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

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

    输入文件有若干组数据,每组数据的第一行是一个正整数n,表示工地的隧道数,接下来的n行每行是用空格隔开的两个整数a和b,表示挖煤点a与挖煤点b由隧道直接连接。输入数据以0结尾。
  • 输出格式

    输入文件中有多少组数据,输出文件中就有多少行。每行对应一组输入数据的结果。

    其中第i行以 Case i: 开始(注意大小写,Case 与 i 之间有空格,i 与 : 之间无空格,: 之后有空格),其后是用空格隔开的两个正整数,第一个正整数表示对于第i组输入数据至少需要设置几个救援出口,第二个正整数表示对于第i组输入数据不同最少救援出口的设置方案总数。输出格式参照以下输入输出样例。
  • 样例输入

    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
  • 样例输出

    Case 1: 2 4

    Case 2: 4 1

解析

一个无向图,故他是有几个vDCC组成的,来讨论每一个vDCC。

先考虑一个问题:
割点是一个很重要的点,如果这个这个vDCC里有割点,说明他可以通过这个割点前往别的vDCC里找出口。

但如果这个割点被砸了,就另外需要安装一个出口跑。

  • \(情况1:\)

    在一个vDCC里不存在割点,则需安装2个出口。

    \(why?\)

    若其中一个出口被砸了,需要另一个出口跑。

  • \(情况2:\)

    在一个vDCC中存在一个割点,则需安装1个出口。

    \(why?\)

    前面说过:\(割点\)是一个很重要的点,如果这个这个vDCC里有割点,说明他可以通过这个割点前往别的vDCC里找出口。

    但如果这个割点被砸了,就另外需要安装一个出口跑。

  • \(情况3:\)

    若一个vDCC里有2个及以上个割点,就什么也不用装了。

    比较好理解,参考前面的解释,这个割点被砸了他还有另一个割点可以跑呢。



    \(\Large{参考:}\)

    点双通分量(vDCC)

    割点
  • \(如何实现:\)

    ans1(出口个数)上述已解决,那么ans2(方案数)用C这个东西解决一下就可以了。

    • 情况1:\(ans1+=2,ans2+=C(2,n)(=n*(n-1)/2)\);(n指点双内点的个数,n个点里选2个)
    • 情况2:\(ans1++,ans2+=C(1,n-1)(=n-1)\);(除去这个割点,剩下n-1个点里选1个)
    • 情况3:什么都不用做。

代码实现

#include<bits/stdc++.h>
#define int long long
#define endl '\n'
using namespace std;
const int N=501;
int dfn[N],low[N],n,ans1,ans2,tot,a,b,num,t;
bool cut[N];
vector<int>e[N],dcc[N];
stack<int>s;
template<typename Tp> inline void read(Tp&x)
{
x=0;register bool f=1;
register char c=getchar();
for(;c<'0'||c>'9';c=getchar()) if(c=='-') f=0;
for(;'0'<=c&&c<='9';c=getchar()) x=(x<<1)+(x<<3)+(c^48);
x=(f?x:~x+1);
}
void clean()
{
memset(dfn,0,sizeof(dfn)),
memset(low,0,sizeof(low)),
memset(cut,0,sizeof(cut)),
memset(e,0,sizeof(e)),
memset(dcc,0,sizeof(dcc)),
ans1=tot=num=0,ans2=1;
}
void tarjan(int x,int fa)
{
dfn[x]=low[x]=++tot;
s.push(x);
int child=0;
if(x==fa&&e[x].size()==0)
{
dcc[++num].push_back(x);
return ;
}
for(int y:e[x])
if(!dfn[y])
{
tarjan(y,fa);
low[x]=min(low[x],low[y]);
if(low[y]>=dfn[x])
{
++num,++child;
if(x!=fa||child>1) cut[x]=1;
int z;
while(z!=y)
z=s.top(),
s.pop(),
dcc[num].push_back(z);
dcc[num].push_back(x);
}
}
else low[x]=min(low[x],dfn[y]);
}
signed main()
{
#ifndef ONLINE_JUDGE
freopen("in.txt","r",stdin);
freopen("out.txt","w",stdout);
#endif
while(1)
{
read(n);
if(n==0) return 0;
clean();
for(int i=1;i<=n;i++)
read(a),read(b),
e[a].push_back(b),
e[b].push_back(a);
for(int i=1;i<=n;i++) if(!dfn[i]) tarjan(i,i);
for(int i=1;i<=num;i++)
{
int tge=0,l=dcc[i].size();
if(l==1) continue;//他可能是森林qwq
for(int j=0;j<l;j++) if(cut[dcc[i][j]]) tge++;
if(tge==0) ans1+=2,ans2*=l*(l-1)/2;
else if(tge==1) ans1++,ans2*=l-1;
}
cout<<"Case "<<++t<<": "<<ans1<<' '<<ans2<<endl;
}
}

重要注意

数据可能是森林,及他是存在孤点的(没有爸爸也没有儿子的孤儿),非常的可怜,如果他这个点双被砸了(这个点双里只有他这一个可怜的孩子),那么他已经死了 \(\Large{qwq}\) ,没有去拯救他的必要了,所以孤点不需要安装出口,需特殊考虑。



右边这个小东西就是孤点,与左边这个大家伙构成了一个森林。

hszxoj 矿场搭建 [tarjan]的更多相关文章

  1. 【BZOJ-2730】矿场搭建 Tarjan 双连通分量

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

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

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

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

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

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

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

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

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

  6. P3225 [HNOI2012]矿场搭建 tarjan割点

    这个题需要发现一点规律,就是先按割点求块,然后求每个联通块中有几个割点,假如没有割点,则需要建两个出口,如果一个割点,则需要建一个出口,2个以上不用建. 题干: 题目描述 煤矿工地可以看成是由隧道连接 ...

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

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

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

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

  9. Tarjan 点双+割点+DFS【洛谷P3225】 [HNOI2012]矿场搭建

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

  10. bzoj2730矿场搭建(Tarjan割点)

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

随机推荐

  1. 精选版:用Java扩展Nginx(nginx-clojure 入门)

    欢迎访问我的GitHub 这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos 本篇概览 今天咱们以Java程序员的视角,来聊聊如何用 ...

  2. 内网DNS解析☞dnsmasq

    内网DNS解析☞dnsmasq 目录 内网DNS解析☞dnsmasq 简介: 安装dnsmasq 问题: 1.怎么让172.30.1.* 与172.30.2.* 两个网段能互相访问? 2.firewa ...

  3. Mac SpringBoot项目 Gradle 7.3 转 Maven 手把手教学,包学会~

    导读 最近我手上有个使用Gradle构建的项目,国内使用Gradle的人相对较少.而且我也觉得Gradle的依赖管理方式有些复杂,让我感到有些困惑.因此,我想将项目转换为Maven构建方式.Maven ...

  4. HTML一键打包APK工具1.9.5更新,新增一机一码功能

    HMTL网址打包APK,可以把本地HTML项目, Egret游戏,网页游戏,或者网站打包为一个安卓应用APK文件,无需编写任何代码,也无需配置安卓开发环境,支持在最新的安卓设备上安装运行. 打包软件会 ...

  5. .NET 8 的 green thread 异步模型被搁置了

    .NET 平台上的green thread 异步模型实验结果最近出来了,具体参见:https://github.com/dotnet/runtimelab/issues/2398 ,实验结果总结一下就 ...

  6. Centos7使用s3fs-fuse挂载minio对象存储实践

    Centos7使用s3fs-fuse挂载minio对象存储实践 事前准备 主机可以访问到对象存储API.例如minio默认的9000端口 主机安装好s3fs软件 已在minio上创建存储桶 安装s3f ...

  7. Java新特性中的Preview功能如何运行和调试

    在每个Java新版本发布的特性中,都会包含一些Preview(预览)功能,这些功能主要用来给开发者体验并收集建议.所以,Preview阶段的功能并不是默认开启的. 如果想体验某个Java版本中的Pre ...

  8. 一种对数据库友好的GUID的变种使用方法

    概述 .NET生成的GUID唯一性很好,用之方便,但是,缺少像雪花算法那样的有序性.虽然分布式系统中做不到绝对的有序,但是,相对的有序对于目前数据库而言,索引效率等方面的提升还是有明显效果的(当然,我 ...

  9. 记一次 .NET某账本软件 非托管泄露分析

    一:背景 1. 讲故事 中秋国庆长假结束,哈哈,在老家拍了很多的短视频,有兴趣的可以上B站观看:https://space.bilibili.com/409524162 ,今天继续给大家分享各种奇奇怪 ...

  10. Dash 2.14版本开始支持动态回调注册!

    本文示例代码已上传至我的Github仓库https://github.com/CNFeffery/dash-master 大家好我是费老师,就在昨晚,Dash框架发布了其2.14.0新版本,新增的功能 ...