并查集--CSUOJ 1601 War
并查集的经典题目:
CSUOJ 1601: War
Time Limit: 1 Sec Memory Limit: 128 MB
Submit:
247 Solved: 70
[Submit][Status][Web
Board]
Description
AME decided to destroy CH’s country. In CH’ country, There are N villages,
which are numbered from 1 to N. We say two village A and B are connected, if and
only if there is a road between A and B, or there exists a village C such that
there is a road between A and C, and C and B are connected. To defend the
country from the attack of AME, CH has decided to build some roads between some
villages. Let us say that two villages belong to the same garrison area if they
are connected.
Now AME has already worked out the overall plan including
which road and in which order would be attacked and destroyed. CH wants to know
the number of garrison areas in his country after each of AME’s attack.
Input
The first line contains two integers N and M — the number of villages and
roads, (2 ≤ N ≤ 100000; 1 ≤ M ≤ 100000). Each of the next M lines contains two
different integers u, v (1<=u, v<=N)—which means there is a road between u
and v. The next line contains an integer Q which denotes the quantity of roads
AME wants to destroy (1 ≤ Q ≤ M). The last line contains a series of numbers
each of which denoting a road as its order of appearance — different integers
separated by spaces.
Output
Output Q integers — the number of garrison areas in CH’s country after each
of AME's attack. Each pair of numbers are separated by a single space.
Sample Input
3 1
1 2
1
1
4 4
1 2
2 3
1 3
3 4
3
2 4 3
Sample Output
3
1 2 3
/*题目大意:n个点m条边(边1,边2...边m),q个要摧毁的边,求按顺序每摧毁一条边后图中连通块的个数 题目分析:并查集,先找出最后状态,反向加边,先将q条路全部摧毁后的连通块个数求出来,然后加边即可,每加一条边,用并查集判断,若两点不在同一连通块中,则合并且连通块个数减1*/
#include<cstdio>
#include<cstring>
#define N 100100
struct Edge {
int u,v;
}edge[N];
bool visited[N];
int a[N],stack[N];
int n,m,q,ans=;
int father[N];
int find(int x)
{
return (father[x]==x)?father[x]:father[x]=find(father[x]);/*注意路径压缩和返回father[x]*/
}
void add_tu()
{
for(int i=;i<=n;++i)
father[i]=i;
for(int i=;i<=m;++i)
{
if(visited[i]) continue;
int r1=find(edge[i].u);
int r2=find(edge[i].v);
if(r1!=r2)
{
father[r2]=r1;
ans--;/*先把连通块的数目设为n,建图的过程中,边建边删除,也可以统计father[i]==i*/
}
}
stack[q]=ans;/*当前的状态是q的状态,不能加边了*/
}
void add_edge()
{
for(int i=q-;i>=;--i)
{
int r1=find(edge[a[i+]].u);/*注意加的这一条边是当前状态的后一条边,因为这个后一条边没有删除,而不是a[i],之前深受其害,连样例都过了*/
int r2=find(edge[a[i+]].v);
if(r1!=r2)
{
father[r2]=r1;
ans--; }
stack[i]=ans;/*注意不要把这句放到if里面*/
}
}
int main()
{
while(scanf("%d%d",&n,&m)!=EOF)/*注意这里不能只用scanf("%d%d",&n,&m),因为这样没法用^z结束程序*/
memset(edge,,sizeof(edge));
memset(visited,false,sizeof(visited));/*别忘记初始化*/
memset(a,,sizeof(a));
memset(stack,,sizeof(stack));
ans=n;
for(int i=;i<=m;++i)
{
scanf("%d%d",&edge[i].u,&edge[i].v);
}
scanf("%d",&q);
for(int i=;i<=q;++i)
{
scanf("%d",&a[i]);
visited[a[i]]=true;/*把不访问的边设上标记*/
}
add_tu();
add_edge();
for(int i=;i<=q-;++i)
printf("%d ",stack[i]);/*CSUOJ这个坑爹的OJ,多输出一个空格,就格式不对*/
printf("%d\n",stack[q]);
}
return ;
}
并查集--CSUOJ 1601 War的更多相关文章
- ZOJ 3261 - Connections in Galaxy War ,并查集删边
In order to strengthen the defense ability, many stars in galaxy allied together and built many bidi ...
- Connections in Galaxy War (逆向并查集)题解
Connections in Galaxy War In order to strengthen the defense ability, many stars in galaxy allied to ...
- ZOJ3261:Connections in Galaxy War(逆向并查集)
Connections in Galaxy War Time Limit: 3 Seconds Memory Limit: 32768 KB 题目链接:http://acm.zju.edu. ...
- 题解报告:zoj 3261 Connections in Galaxy War(离线并查集)
Description In order to strengthen the defense ability, many stars in galaxy allied together and bui ...
- ZOJ3261 Connections in Galaxy War —— 反向并查集
题目链接:https://vjudge.net/problem/ZOJ-3261 In order to strengthen the defense ability, many stars in g ...
- ZOJ 3261 Connections in Galaxy War(逆向并查集)
参考链接: http://www.cppblog.com/yuan1028/archive/2011/02/13/139990.html http://blog.csdn.net/roney_win/ ...
- ZOJ 3261 Connections in Galaxy War (逆向+带权并查集)
题意:有N个星球,每个星球有自己的武力值.星球之间有M条无向边,连通的两个点可以相互呼叫支援,前提是对方的武力值要大于自己.当武力值最大的伙伴有多个时,选择编号最小的.有Q次操作,destroy为切断 ...
- UVA 10158 War(并查集)
//思路详见课本 P 214 页 思路:直接用并查集,set [ k ] 存 k 的朋友所在集合的代表元素,set [ k + n ] 存 k 的敌人 所在集合的代表元素. #include< ...
- ZOJ-3261 Connections in Galaxy War 并查集 离线操作
题目链接:https://cn.vjudge.net/problem/ZOJ-3261 题意 有n个星星,之间有m条边 现一边询问与x星连通的最大星的编号,一边拆开一些边 思路 一开始是真不会,甚至想 ...
随机推荐
- tf.segment_sum和tf.unsorted_segment_sum理解实例
本文来自 guotong1988 的CSDN 博客 ,全文地址请点击:https://blog.csdn.net/guotong1988/article/details/77622790 import ...
- VueJS 轻松支持 JSX 配置
使用: babel-preset-vue-app TODO
- JS 判断是否是微信浏览器 webview
原理很简单,就是判断 ua 中是否有字段 “micromessenger" 代码如下: function isWechat () { var ua = window.navigator.us ...
- 74.VS2013和opencv3.1.0安装教程
一.先下载文件 1.VS2013 VS2013有很多版本,专业版,旗舰版,中文英文之类的,所对应的密钥也不一样.我选择的是简体中文专业版.下载链接如下. http://www.musnow.com/t ...
- ldconfig是一个动态链接库管理命令
ldconfig是一个动态链接库管理命令 为了让动态链接库为系统所共享,还需运行动态链接库的管理命令--ldconfig ldconfig 命令的用途,主要是在默认搜寻目录(/lib和/usr/li ...
- 有关mysql的innodb_flush_log_at_trx_commit参数【转】
一.参数解释 0:log buffer将每秒一次地写入log file中,并且log file的flush(刷到磁盘)操作同时进行.该模式下在事务提交的时候,不会主动触发写入磁盘的操作. 1:每次事务 ...
- IO的学习与使用
一.IO的学习方法:IO中包含了很多的类,推荐的学习方式是:“举一反三,掌握一种,以此类推”. 二.I/O操作的目标: 输入:从数据源(在数据源和程序之间建立的一个数据流淌的“管道”)中读取数据(文件 ...
- python 内置函数eval()、exec()、compile()
eval 函数的作用: 计算指定表达式的值.也就是说它要执行的python代码只能是单个表达式,而不是复杂的代码逻辑. eval(source, globals=None, locals=Non ...
- Android Debug Bridge (adb) device - no permissions
I have a problem connecting HTC Wildfire A3333 in debugging mode with my Fedora Linux 17. Adb says ...
- CSS3绘图与动画
<meta http-equiv="X-UA-Compatible" content="ie=edge"> 1.CSS3-2D转换属性:trans ...