点此看题面

大致题意: 一张无向图,要求你在去掉任意一个节点之后,剩余的每个节点都能到达一个救援出口,问至少需要几个救援出口。

第一步:\(Tarjan\)求割点

首先,我们要跑一遍\(Tarjan\)来求出割点

求完割点后,这样我们就能求出原图除了割点以外其他点所形成的若干个联通块(相当于把割点去掉之后形成的联通块)。

第二步:分类讨论

接下来,我们要对上面求出来的每一个联通块相邻的割点个数进行分类讨论:

  • 如果当前联通块相邻的割点个数为0

    显然对于这个联通块我们需要造2个救援出口,不然万一一个救援出口崩塌了,就出不去了。

    而方案数就相当于\(C_{Size}^2\),即\(\frac{Size(Size-1)}2\)。

  • 如果当前联通块相邻的割点个数为1

    那么对于这个联通块我们只需要造一个救援出口,因为就算救援出口崩塌了,还可以通过割点去另一个联通块内的救援出口。

    而方案数就是\(Size\)。

  • 如果当前联通块相邻的割点个数大于等于2

    那么对于这个联通块我们就不需要造救援出口了,因为就算某个割点崩塌了,我们依然可以通过另一个割点到达其它联通块的救援出口。

这样代码实现就不难了。

代码

#include<bits/stdc++.h>
#define max(x,y) ((x)>(y)?(x):(y))
#define min(x,y) ((x)<(y)?(x):(y))
#define abs(x) ((x)<0?-(x):(x))
#define LL long long
#define ull unsigned long long
#define swap(x,y) (x^=y,y^=x,x^=y)
#define tc() (A==B&&(B=(A=ff)+fread(ff,1,100000,stdin),A==B)?EOF:*A++)
#define pc(ch) (pp_<100000?pp[pp_++]=ch:(fwrite(pp,1,100000,stdout),pp[(pp_=0)++]=ch))
#define N 500
#define add(x,y) (e[++ee].nxt=lnk[x],e[lnk[x]=ee].to=y)
int pp_=0;char ff[100000],*A=ff,*B=ff,pp[100000];
using namespace std;
int n,m,ans1=0,ee=0,cnt=0,d=0,top=0,Exist[N+5],lnk[N+5],dfn[N+5],low[N+5],vis[N+5],IsCut[N+5],used[N+5],Size[N+5],tot[N+5],Stack[N+5];
ull ans2=1;
struct edge
{
int to,nxt;
}e[2*N+5];
inline void read(int &x)
{
x=0;static char ch;
while(!isdigit(ch=tc()));
while(x=(x<<3)+(x<<1)+ch-48,isdigit(ch=tc()));
}
inline void write(ull x)
{
if(x>9) write(x/10);
pc(x%10+'0');
}
inline void Tarjan(int x,int lst)//用Tarjan找割点
{
register int i,tot=0;
for(dfn[x]=low[x]=++d,i=lnk[x];i;i=e[i].nxt)
{
if(!(e[i].to^lst)) continue;
if(!dfn[e[i].to])
{
Tarjan(e[i].to,x),low[x]=min(low[x],low[e[i].to]),++tot;
if(lst&&low[e[i].to]>=dfn[x]) IsCut[x]=1;
}
else low[x]=min(low[x],dfn[e[i].to]);
}
if(!lst&&tot>=2) IsCut[x]=1;
}
inline void dfs(int x)//dfs遍历除去割点后的一个联通块
{
register int i,v;
for(Size[x]=vis[x]=1,tot[x]=0,i=lnk[x];i;i=e[i].nxt)//枚举每一个相邻的节点
{
if(IsCut[v=e[i].to])//如果这个节点是割点
{
if(!used[v]) used[Stack[++top]=v]=1,++tot[x];//如果这个割点没有访问过,就标记这个割点已访问,并将这个联通块相邻的割点个数加1
continue;//跳过
}
if(!vis[v]) dfs(v),Size[x]+=Size[v],tot[x]+=tot[v];//如果这个节点没被访问过,就去访问这个节点,并更新当前节点信息
}
}
int main()
{
register int i,j,x,y,T=0;
while(read(n),n)
{
for(ans1=ee=0,ans2=i=1;i<=n+1;++i) lnk[i]=dfn[i]=low[i]=IsCut[i]=Exist[i]=vis[i]=0;//初始化,将数组清空
for(i=1;i<=n;++i) read(x),read(y),add(x,y),add(y,x),Exist[x]=Exist[y]=1;
for(i=1;i<=n+1;++i) if(Exist[i]&&!dfn[i]) Tarjan(i,0);//Tarjan求割点
for(i=1;i<=n+1;++i)
{
if(!Exist[i]||IsCut[i]||vis[i]) continue;//如果这个节点不存在,或这个节点是割点,或这个节点已经被访问过,就跳过
top=0,dfs(i);//dfs遍历这个联通块
while(top) used[Stack[top--]]=0;//将访问过的割点标记为未访问
if(!tot[i]) ans1+=2,ans2*=1LL*Size[i]*(Size[i]-1)>>1;//如果这个联通块相邻的割点数为0,就需要加两个救援出口,方案数为Size(Size-1)/2
else if(tot[i]==1) ++ans1,ans2*=Size[i]; //如果这个联通块相邻的割点数为1,就需要加一个救援出口,方案数为Size
//如果这个联通块相邻的割点数大于等于2,就不需要加救援出口了
}
pc('C'),pc('a'),pc('s'),pc('e'),pc(' '),write(++T),pc(':'),pc(' '),write(ans1),pc(' '),write(ans2),pc('\n');//输出答案
}
return fwrite(pp,1,pp_,stdout),0;
}

【BZOJ2730】[HNOI2012] 矿场搭建(找割点)的更多相关文章

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

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

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

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

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

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

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

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

  5. [HNOI2012]矿场搭建(割点)

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

  6. [BZOJ2730][HNOI2012]矿场搭建(求割点)

    题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=2730 分析: 如果坍塌的点不是割点,那没什么影响,主要考虑坍塌的点是割点的情况. 显然 ...

  7. BZOJ2730:[HNOI2012]矿场搭建(双连通分量)

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

  8. BZOJ2730 [HNOI2012]矿场搭建[点双连通分量]

    看到删去一个点,需要剩下的都和关键点连通,有端联想到找点双,因为他怎么删点都是连通的. 对于一个孤立的点双,至少要设两个关键点. 如果两个点双以一个割点连接,假设断掉这个割点,两个块至少要各设一个关键 ...

  9. BZOJ2730: [HNOI2012]矿场搭建

    传送门 图的连通性相关的必和割点割边之类的有关. 题目要求对于一个无向图,任意一点被删除后,所有点都和某些指定点是联通的. 这道题比较简单的做法就是求出来所有的块.对于一个块,如果块里有两个及两个以上 ...

  10. BZOJ2730——[HNOI2012]矿场搭建

    bzoj2730 & world final 2011 H 1.题目大意:就是有一个无向图,让你在里面选择点,使得,无论哪个点没了以后,其他的点都能到达你选择的任何一个点,输出最少 选择几个点 ...

随机推荐

  1. 移动web 的viewport设置注意事项的详细解释 六一快乐=-_-_

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  2. Linux调优(网络)

    定义socket接受缓冲大小 net.core.rmem_default = N #接受 net.core.rmem_max = N net.core.wmem_default = N #发送 net ...

  3. EventLoop-浏览器与Node.js--整理

    近来面试中会遇到的问题,关于浏览器和Nodejs两个运行环境的Event loop. 整理值得阅读的优秀文章 参考文章: 1.不要混淆nodejs和浏览器的eventloop 2.nodejs官网关于 ...

  4. webpack配置Jquery全局包及全局包插件

    一:在配置文件配置: plugins: [ //将来以template为模版,生成一个index.html并且发布到webpack-dev-server开启的node服务器上面去 new HtmlWe ...

  5. gulp前端自动化构建工具

    博主不易,不求赞赏,希望把自己遇到的难点写出来,以及希望自己能有能力写出一篇不错的博文. 前端构建工具本人 bootstrap+jquery用gulp vue+element 用webpack 引文 ...

  6. linux下find查找与批量替换文件中指定内容

    经常在部署tomcat时需要替换配置文件中的ip,find命令批量替换还是很方便的 查找需要替换的ip,看看哪些文件有配置这个ip,执行下面命令: find ./ -type f -regex &qu ...

  7. BZOJ 4165 矩阵 堆

    先把每个长为$mina$,宽为$minb$的矩阵扔到堆里,然后由于矩阵中的数都是正的,所以我们每取出来一个矩形,,就把他向四个方向扩张一行,再把这新的且更大的四个矩形扔到堆里.注意判重,于是我比较懒用 ...

  8. ACM-ICPC 2018 沈阳赛区网络预赛 D. Made In Heaven(约束第K短路)

    题意:求11到nn的第kk短的路径长度,如果超过TT输出Whitesnake!Whitesnake!,否则输出yareyaredawayareyaredawa. 好无以为 , 这就是一道模板题, 当是 ...

  9. Django组件-cookie,session

    昨日内容回顾: json 轻量级的数据交换格式 在python 序列化方法:json.dumps() 反序列化方法:json.loads() 在JS中: 序列化方法:JSON.stringfy() 反 ...

  10. 提高mysql千万级大数据SQL查询优化30条经验

    1.对查询进行优化,应尽量避免全表扫描,首先应考虑在 where 及 order by 涉及的列上建立索引. 2.应尽量避免在 where 子句中对字段进行 null 值判断,否则将导致引擎放弃使用索 ...