并查集--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星连通的最大星的编号,一边拆开一些边 思路 一开始是真不会,甚至想 ...
随机推荐
- MSSQL 基础知识002
---启用sa账号 1. 先使用一个windows账号登陆. 2.在数据库实例上面右键,属性,安全性,登录名,sa. 右键,属性. 常规,修改sa的密码. 状态,启用sa账号. 主键的作用: 1.唯一 ...
- three.js_camera相机
https://blog.csdn.net/yangnianbing110/article/details/51275927 文章地址
- tensorflow.nn.bidirectional_dynamic_rnn()函数的用法
在分析Attention-over-attention源码过程中,对于tensorflow.nn.bidirectional_dynamic_rnn()函数的总结: 首先来看一下,函数: def bi ...
- openjudge-NOI 2.6-1768 最大子矩阵
题目链接:http://noi.openjudge.cn/ch0206/1768/ 题解: 如果用O(n4)的算法肯定会炸,需要压缩掉一维的空间,只需要简单加和就好啦 例如,我们要对样例中第2-4行D ...
- 5.Python3标准库-日期和时间
''' 不同于int,str,float,Python没有包含对应日期和时间的原生类型,不过提供了3个相应的模块,可以采用多种表示来管理日期和时间值 time模块由底层C库提供与时间相关的函数.它包含 ...
- java基础9 main函数、this、static、super、final、instanceof 关键字
一.main函数详解 1.public:公共的.权限是最大的,在任何情况都可以访问 原因:为了保证jvm在任何情况下都可以访问到main法2.static:静态,静态可以让jvm调用更方便,不需要用 ...
- maven实战系列
Maven实战(一)安装和配置 Maven实战(二)构建简单Maven项目 Maven实战(三)Eclipse构建Maven项目 Maven实战(四)生命周期 Maven实战(五)坐标详解 Maven ...
- Fiddler Web Session 列表(1)
Web Session 列表 位置: Web Session 列表 位于Fiddler界面的左侧 ,是Fiddler所抓取到的所有Session会话的列表集合. Web Session 列表 栏名词解 ...
- hdu 1847(SG函数,巴什博弈)
Good Luck in CET-4 Everybody! Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K ...
- awk进阶
整理的awk的小技巧 begin是要放在正则前面的,按照这个顺序: awk 'begin{} /.*?/ {action}end{}' file FS=':' 和 -F: 是等同的 -F 表示以 XX ...