题目描述

in English:
Farmer John and his cows are planning to leave town for a long vacation, and so FJ wants to temporarily close down his farm to save money in the meantime.

The farm consists of NN barns connected with MM bidirectional paths between some pairs of barns (1 \leq N, M \leq 30001≤N,M≤3000 ). To shut the farm down, FJ plans to close one barn at a time. When a barn closes, all paths adjacent to that barn also close, and can no longer be used.

FJ is interested in knowing at each point in time (initially, and after each closing) whether his farm is “fully connected” – meaning that it is possible to travel from any open barn to any other open barn along an appropriate series of paths. Since FJ’s farm is initially in somewhat in a state of disrepair, it may not even start out fully connected.

in Chinese:
FJ和他的奶牛们正在计划离开小镇做一次长的旅行,同时FJ想临时地关掉他的农场以节省一些金钱。

这个农场一共有被用M条双向道路连接的N个谷仓(1<=N,M<=3000)。为了关闭整个农场,FJ 计划每一次关闭掉一个谷仓。当一个谷仓被关闭了,所有的连接到这个谷仓的道路都会被关闭,而且再也不能够被使用。

FJ现在正感兴趣于知道在每一个时间(这里的“时间”指在每一次关闭谷仓之前的时间)时他的农场是否是“全连通的”——也就是说从任意的一个开着的谷仓开始,能够到达另外的一个谷仓。注意自从某一个时间之后,可能整个农场都开始不会是“全连通的”。

说白了就是:

有N个农场,编号为1到N,有M条双向公路连接,其中1≤N,M≤200000,每次封锁一个农场,
当一个农场被封闭时,和该农场连接的道路全部断开,不能再通行。现在给出一个封锁顺序,
请计算每次封锁一个农场后,剩余的未封锁的农场是否连通。

输入输出格式

in English:

输入格式:
The first line of input contains NN and MM . The next MM lines each describe a

path in terms of the pair of barns it connects (barns are conveniently numbered

1 \ldots N1…N ). The final NN lines give a permutation of 1 \ldots N1…N

describing the order in which the barns will be closed.

输出格式:
The output consists of NN lines, each containing “YES” or “NO”. The first line

indicates whether the initial farm is fully connected, and line i+1i+1 indicates

whether the farm is fully connected after the ii th closing.

in Chinese:

Input Data
第一行,包含两个整数N和M。
接下来M行,每行两个整数,描述一条道路。
接下来N行,每行一个整数,依次表示封锁的农场编号。

Output Data
输出包含N行,每行一个字符串“YES”或者“NO”,表示连通或者不连通。
其中第i行,表示第i个农场未被封锁时,所有未被封锁的农场的连通情况。
第i+1行,表示第i个农场被封锁后,所有未被封锁的农场的连通情况。
特别地,第1行表示所有农场未被封锁前,所有农场的连通情况。

输入输出样例

输入样例#1:
4 3
1 2
2 3
3 4
3
4
1
2
输出样例#1:
YES
NO
YES

然后我看着这道题就码出了第一个程序:

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<vector>using namespace std;
int n,m,f[210000];
bool close[210000],t[210000];
vector <int> pat[210000];int fa(int x)
{if(f[x]==x) return x;return f[x]=fa(f[x]);
}bool check()
{bool one=false;for(int i=1;i<=n;++i)f[i]=i;for(int i=1;i<=n;++i) if(!close[i])for(int j=0;j<pat[i].size();++j){if(close[pat[i][j]]) continue;int fx=fa(i),fy=fa(pat[i][j]);if(fx!=fy) f[fx]=fy;}  /*然后这里判断连通图的个数的方法也很朴素, 就是查祖先的个数是否超过了1,就酱*/for(int i=1;i<=n;++i)if(!close[i] && f[i]==i){if(!one) one=true;else return false;}return true;  //通过了检测之后直接返回TRUE(表示联通)
}int main()
{scanf("%d%d",&n,&m);for(int i=1;i<=m;++i)  //就是记录了一下每个点的联通边{int from,to;scanf("%d%d",&from,&to);pat[from].push_back(to);pat[to].push_back(from);}if(check()) printf("YES\n"); //先是没删点的时候跑一遍checkelse printf("NO\n");for(int i=1;i<n;++i)//然后这里就是按顺序来,读一个点,删一个点,再看图是否联通{int x; scanf("%d",&x);close[x]=1;if(check()) printf("YES\n");else printf("NO\n");}return 0;
}

然后我惊奇的发现洛谷上A了,这数据是真心水,这样也能水过去(吸口氧貌似还能再降三分之二的时间)。

然后看到这里哭了T_T

换个地儿评测TLE!还能不能愉快的玩耍了?

行,这数据真心强!我还真不信就倒在这儿了_ (:зゝ∠) _
于是乎我就换了个方法:做个倒过来的并查集。

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<vector>using namespace std;
int n,m,f[210000],add[210000];
bool open[210000],t[210000];
vector <int> pat[210000];int read(){int x=0; char c=getchar();while(!isdigit(c)) c=getchar();while(isdigit(c)) x=x*10+c-'0',c=getchar();return x;
}int fa(int x)
{if(f[x]==x) return x;return f[x]=fa(f[x]);
}bool check(int x)
{bool one=false;for(int i=0;i<pat[x].size();++i) //没什么两样,就是把能和新开张的农场相连的农场合并了一下{if(!open[pat[x][i]]) continue;int fx=fa(x),fy=fa(pat[x][i]);if(fx!=fy) f[fx]=fy;}for(int i=1;i<=n;++i) //判断还是老办法(TLE最根本的原因也是在这里,这个等会儿讲)if(open[i] && f[i]==i){if(!one) one=true;else return false;}return true;
}int main()
{n=read();m=read();for(int i=1;i<=m;++i){int from=read(),to=read();pat[from].push_back(to);pat[to].push_back(from);}for(int i=1;i<=n;++i)  //并查集的预处理就直接放在主函数里了f[i]=i;for(int i=1;i<=n;++i)  //然后实现倒着把数据读入add[n-i]=read();open[add[0]]=1;  //表示第一家农场开门t[n]=1;for(int i=1;i<n;++i){open[add[i]]=1; //然后一家接着一家开门(抢生意)if(check(add[i]))  t[n-i]=1;  //如果check出来图联通了,就(反着)标记t为1}for(int i=1;i<=n;++i)  //然后按读入的顺序输出if(t[i]) printf("YES\n");else printf("NO\n");return 0;
}

然后洛谷上一交,心里美滋滋:

时间降了十倍左右!这下还会爆就真是。。。额

兵败如山倒= =

_ ( :зゝ∠)_这次是真爬不起来了QAQ

然而我最后还是爬起来了_ ( :зゝ∠)_ (其实也就是做了个判断优化 = =|||)

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<vector>using namespace std;
int n,m,ans,f[210000],add[210000];
bool open[210000],t[210000];
vector <int> pat[210000];int read(){int x=0; char c=getchar();while(!isdigit(c)) c=getchar();while(isdigit(c)) x=x*10+c-'0',c=getchar();return x;
}int fa(int x)
{if(f[x]==x) return x;return f[x]=fa(f[x]);
}bool check(int x)
{++ans; //其他我都不想介绍了,主要也就是这里。。。//ans++代表多了一个图(新开张的农场自为一个图)for(int i=0;i<pat[x].size();++i){if(!open[pat[x][i]]) continue;int fx=fa(x),fy=fa(pat[x][i]);if(fx!=fy) f[fx]=fy,ans--;//然后在这里进行一个减图操作,如果两个集合合并了,图必然--,然后ans就--了}if(ans) //最后判断一下多余的图是否为0,是的话就返回FALSE就这么easy(卡了我60)return false;return true;
}int main()
{n=read();m=read();for(int i=1;i<=m;++i){int from=read(),to=read();pat[from].push_back(to);pat[to].push_back(from);}for(int i=1;i<=n;++i)f[i]=i;for(int i=1;i<=n;++i)add[n-i]=read();open[add[0]]=1;for(int i=1;i<n;++i){open[add[i]]=1;if(check(add[i]))  t[n-i]=1;}for(int i=1;i<n;++i)if(t[i]) printf("YES\n");else printf("NO\n");printf("YES\n");return 0;
}

然后洛谷上又A了一遍,时间又降了一半的样子(虽说A了还是没有什么喜悦的赶脚,毕竟连跪了两把T_T)

然后又去挑战了一下BOSS:

终于A了!_ (||Xゝ∠)_

然后我就这样跪着(不如说是躺着)过了一道(巨)坑。。。 _ (:зゝ∠)_

然后本blog就结束了,童鞋们下次见!

转载于:https://www.cnblogs.com/Judge/p/9383041.html

一道并查集的(坑)题:关闭农场closing the farm相关推荐

  1. 洛谷P3144 [USACO16OPEN]关闭农场Closing the Farm_Silver

    洛谷P3144 [USACO16OPEN]关闭农场Closing the Farm_Silver 题目描述 FJ和他的奶牛们正在计划离开小镇做一次长的旅行,同时FJ想临时地关掉他的农场以节省一些金钱. ...

  2. CodeForces - 123A prime permutation(并查集,水题)

    题目链接:点击查看 题目大意:给出一个字符串s,问能否通过重组其字母顺序,从而满足:若字符串下标从1开始,对于每一个质数下标,满足 题目分析:其实在纸上稍微写写画画就能看出个大概的规律,当字符串的长度 ...

  3. All men are brothers(并查集+思维 好题!!!)

    链接:https://ac.nowcoder.com/acm/contest/889/E 来源:牛客网 Amy asks Mr. B problem E. Please help Mr. B to s ...

  4. 【CSU - 1004】Xi and Bo(并查集,裸题)

    题干: Bo has been in Changsha for four years. However he spends most of his time staying his small dor ...

  5. Hrbust 1849 商品中心【贪心+思维+并查集】好题!好题!

    商品中心 Time Limit: 2000 MS Memory Limit: 32768 K Total Submit: 7(5 users) Total Accepted: 3(3 users) R ...

  6. 1114 Family Property (25分) (并查集) 复杂题 经典并查集

    This time, you are supposed to help us collect the data for family-owned property. Given each person ...

  7. 并查集小结 (参考birdfly+修改)

    并查集的作用:并和查,即合并和查找,将一些集合合并,快速查找或判断某两个集合的关系,或某元素与集合的关系,或某两个元素的关系. 并查集的结构:并查集主要操作对象是森林,树的结构赋予它独特的能力,对整个 ...

  8. 【CCCC】L2-024 部落 (25分),,并查集,模板水题,统计集合个数

    problem L2-024 部落 (25分) 在一个社区里,每个人都有自己的小圈子,还可能同时属于很多不同的朋友圈.我们认为朋友的朋友都算在一个部落里,于是要请你统计一下,在一个给定社区中,到底有多 ...

  9. 0x41.数据结构进阶 - 并查集

    目录 一.路径压缩与按秩合并 1.AcWing 237. 程序自动分析(NOIP2015) 二.边带权并查集 1.AcWing 238. 银河英雄传说(边带权并查集模板) 2.AcWing 239. ...

最新文章

  1. 二十六、内存管理的概念
  2. 学生用计算机2ndf,手机计算器2ndf键在哪 等同于INV键
  3. java shp求相交面积_shp文件自相交处理的方法
  4. different styles of button in footer area
  5. [导入].Net 中处理Word(2007)文档的一种方法
  6. java 生成Excel开门篇
  7. FFmpeg学习教程
  8. 百度NLP词 语相似度接口Demo
  9. JS 解决IE浏览器出现Automation 服务器不能创建对象
  10. 电子商务类网站CDN加速方案
  11. python升序和降序排序_sql升序和降序(orderby升序和降序排列显示)
  12. 问学计算机的同学买什么电脑,学生用的笔记本电脑什么样的好
  13. youtube-dl 环境搭建 使用
  14. 4595: [Shoi2015]激光发生器
  15. MySQL函数 NOW()
  16. 新增计算机专硕!数二英二,只有数据结构!合肥学院招80人!
  17. Pr 电影开场帷幕拉开效果和轨道遮罩的应用
  18. 汇编工具DOSBOX安装使用
  19. 图解HTTP读书笔记.第八章
  20. 统计学中的Bootstrap方法(Bootstrap抽样)

热门文章

  1. 图示评审技术_编制清单报价最需要施工技术课程中的哪些知识?
  2. 网站seo优化每天需要做哪些事情呢?
  3. 网站优化中搜索引擎为何重视原创内容呢?
  4. python自动化教程_Python自动化开发实战视频课程-全新基础篇_Python教程
  5. python 为什么没有重载_python是否支持重载
  6. 一年的第几周怎么算_部编版一年级下册第7课《怎么都快乐》图文讲解+知识点梳理...
  7. 开发日记-20190824 关键词 读书笔记《掌控习惯》DAY 3
  8. python 的进程池不可嵌套
  9. 实践浅解free的具体作用(C语言)
  10. WebApp NativeApp HybirdApp