博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
BZOJ1015[JSOI2008]星球大战starwar[并查集]
阅读量:5025 次
发布时间:2019-06-12

本文共 4130 字,大约阅读时间需要 13 分钟。

1015: [JSOI2008]星球大战starwar

Time Limit: 3 Sec  Memory Limit: 162 MB
Submit: 5253  Solved: 2395
[][][]

Description

  很久以前,在一个遥远的星系,一个黑暗的帝国靠着它的超级武器统治者整个星系。某一天,凭着一个偶然的

机遇,一支反抗军摧毁了帝国的超级武器,并攻下了星系中几乎所有的星球。这些星球通过特殊的以太隧道互相直
接或间接地连接。 但好景不长,很快帝国又重新造出了他的超级武器。凭借这超级武器的力量,帝国开始有计划
地摧毁反抗军占领的星球。由于星球的不断被摧毁,两个星球之间的通讯通道也开始不可靠起来。现在,反抗军首
领交给你一个任务:给出原来两个星球之间的以太隧道连通情况以及帝国打击的星球顺序,以尽量快的速度求出每
一次打击之后反抗军占据的星球的连通快的个数。(如果两个星球可以通过现存的以太通道直接或间接地连通,则
这两个星球在同一个连通块中)。

Input

  输入文件第一行包含两个整数,N (1  < =  N  < =  2M) 和M (1  < =  M  < =  200,000),分别表示星球的

数目和以太隧道的数目。星球用 0 ~ N-1的整数编号。接下来的M行,每行包括两个整数X, Y,其中(0 < = X <> 
Y 表示星球x和星球y之间有“以太”隧道,可以直接通讯。接下来的一行为一个整数k,表示将遭受攻击的星球的
数目。接下来的k行,每行有一个整数,按照顺序列出了帝国军的攻击目标。这k个数互不相同,且都在0到n-1的范
围内。

Output

  输出文件的第一行是开始时星球的连通块个数。接下来的N行,每行一个整数,表示经过该次打击后现存星球

的连通块个数。

Sample Input

8 13
0 1
1 6
6 5
5 0
0 6
1 2
2 3
3 4
4 5
7 1
7 2
7 6
3 6
5
1
6
3
5
7

Sample Output

1
1
1
2
3
3

 


 

并查集不支持删除,所以倒着处理,每次加入一个点

用tot记录连通块个数,没加一个点就+1,遇到两个合并就-1,不会被破坏的点也用加的方法

加一个点是要便利它的所有边

/**************************************************************    Problem: 1015    User: thwfhk    Language: C++    Result: Accepted    Time:1284 ms    Memory:13788 kb****************************************************************/ ////  main.cpp//  bzoj1015////  Created by Candy on 9/17/16.//  Copyright © 2016 Candy. All rights reserved.// #include 
#include
#include
#include
using namespace std;const int N=4e5+5,M=2e5+5;int read(){    char c=getchar();int x=0,f=1;    while(c<'0'||c>'9'){
if(c=='-')f=-1; c=getchar();}    while(c>='0'&&c<='9'){x=x*10+c-'0'; c=getchar();}    return x*f;}struct edge{    int v,ne;}e[M<<1];int h[N],cnt=0;inline void ins(int u,int v){    cnt++;    e[cnt].v=v;e[cnt].ne=h[u];h[u]=cnt;    cnt++;    e[cnt].v=u;e[cnt].ne=h[v];h[v]=cnt;} int n,m,x,y,k,tot=0,ans[N];int has[N],lst[N],vis[N];//vis for ins from 0int fa[N];int find(int x){
return x==fa[x]?x:fa[x]=find(fa[x]);}inline void merge(int x,int y){    int f1=find(x),f2=find(y);    if(f1!=f2) fa[f1]=f2,tot--;}void add(int u){    for(int i=h[u];i;i=e[i].ne){        int v=e[i].v;        if(vis[v]) merge(u,v);    }}int main(int argc, const char * argv[]) {    n=read();m=read();    for(int i=1;i<=m;i++){        x=read()+1;y=read()+1;ins(x,y);    }    k=read();    for(int i=1;i<=n;i++) has[i]=1,fa[i]=i;    for(int i=1;i<=k;i++){        x=read()+1;        has[x]=0;        lst[i]=x;    }         for(int u=1;u<=n;u++)        if(has[u]){            tot++;            add(u);            vis[u]=1;        }         ans[k+1]=tot;    for(int i=k;i>=1;i--){        tot++;        add(lst[i]);        vis[lst[i]]=1;        ans[i]=tot;    }    for(int i=1;i<=k+1;i++) printf("%d\n",ans[i]);    return 0;}

 

[2016-11-15]

注意一开始也要求一下cc个数

加入一个点可能连起好多个cc来,用并查集维护连通性

#include 
#include
#include
#include
#include
using namespace std;const int N=4e5+5,M=2e5+5,INF=1e9;inline int read(){ char c=getchar();int x=0,f=1; while(c<'0'||c>'9'){
if(c=='-')f=-1;c=getchar();} while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();} return x*f;}int n,m,k,a[N],des[N],x,y,ans[N];struct edge{ int v,w,ne;}e[M<<1];int h[N],cnt=0;inline void ins(int u,int v){ cnt++; e[cnt].v=v;e[cnt].ne=h[u];h[u]=cnt; cnt++; e[cnt].v=u;e[cnt].ne=h[v];h[v]=cnt;}int fa[N];inline int find(int x){
return x==fa[x]?x:fa[x]=find(fa[x]);}int tot=0;inline void unn(int x,int y){ int f1=find(x),f2=find(y); if(f1!=f2) fa[f1]=f2,tot--;}void add(int u){ for(int i=h[u];i;i=e[i].ne){ int v=e[i].v; if(!des[v]) unn(u,v); }}int main(){ n=read();m=read(); for(int i=1;i<=m;i++){x=read()+1;y=read()+1;ins(x,y);} k=read(); for(int i=1;i<=k;i++) a[i]=read()+1,des[a[i]]=1; for(int i=1;i<=n;i++) fa[i]=i; for(int i=1;i<=n;i++) if(!des[i]){ tot++; add(i); } ans[k+1]=tot; for(int i=k;i>=1;i--){ tot++; des[a[i]]=0; add(a[i]); ans[i]=tot; } for(int i=1;i<=k+1;i++) printf("%d\n",ans[i]);}

 

转载于:https://www.cnblogs.com/candy99/p/5883445.html

你可能感兴趣的文章
第七周作业
查看>>
函数式编程与参数
查看>>
flush caches
查看>>
SSAS使用MDX生成脱机的多维数据集CUB文件
查看>>
ACM_hdu1102最小生成树练习
查看>>
MyBatis源码分析(一)--SqlSessionFactory的生成
查看>>
android中ListView点击和里边按钮或ImageView点击不能同时生效问题解决
查看>>
CTF常用工具之汇总
查看>>
java的面向对象 (2013-09-30-163写的日志迁移
查看>>
HDU 2191 【多重背包】
查看>>
51nod 1433 0和5【数论/九余定理】
查看>>
【AHOI2013复仇】从一道题来看DFS及其优化的一般步骤和数组分层问题【转】
查看>>
less 分页显示文件内容
查看>>
如何对数据按某列进行分层处理
查看>>
[Qt] this application failed to start because it could not find or load the Qt platform plugin
查看>>
Git Submodule管理项目子模块
查看>>
学会和同事相处的30原则
查看>>
NOJ——1568走走走走走啊走(超级入门DP)
查看>>
文件操作
查看>>
Python:GUI之tkinter学习笔记3事件绑定(转载自https://www.cnblogs.com/progor/p/8505599.html)...
查看>>