JSOI2008星球大战

Posted 汉谡

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了JSOI2008星球大战相关的知识,希望对你有一定的参考价值。

题目链接

 

直接算每次破坏会拆开多少个连通块貌似不可做。考虑反着加边用并查集合并。

那么我们首先用$vector$存下每个点出边到的点的序列。注意:$min[1,2 imes 10^5]$而$nin [1,2 imes m]$所以$nin [1,4 imes 10^5]$。

读入破坏的顺序之后标记一下这些即将被破坏的点。

开始处理,首先用$BFS$连一下不会被破坏的点算连通块。注意:别连到了会被破坏的点(已经被标记),处理不好容易算重。

这里最好维护一下并查集要用的$f_i$和$sz_i$。注意:这里不用$DFS$的原因是内存只给了$125M$,处理不好容易爆栈$MLE$。

然后开始反向加边。注意:要求的是$0sim k$时刻结束时的连通块计数,那么反向处理$i$时刻的结果就是原来$i-1$时刻的答案,自然反向加边之前的就是$k$时刻的答案。

时光逆流,每个时刻就有一个被光复的星球,没加边之前是孤立的,先把$cnt++$。

开始逐边恢复该星球的以太通讯,若目标不在同一个连通块内则有两个连通块合并,$cnt--$。注意:别向被破坏的点(有标记)连边。

恢复完毕该星球的以太通讯之后,把它的标记摘除。恭喜反抗军完成光复!【滑稽】

 

代码(100分):

技术图片
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<vector>
#include<queue>
#include<map>
#include<set>
#define IL inline
#define RG register
#define _1 first
#define _2 second
using namespace std;
typedef unsigned int UI;
const int N=4e5;

    int n,m;
    vector<int>g[N+3];
    int k,a[N+3];
    bool p[N+3];
    
    int f[N+3],sz[N+3];
    int cnt;
    int s[N+3],hd,tl;

int find(int x){return (x==f[x])?x:(f[x]=find(f[x]));}

IL void merge(int x,int y){
    if(sz[x]<sz[y])    swap(x,y);
    f[y]=x;    sz[x]+=(int)(sz[x]==sz[y]);
}

    int ans[N+3];

int main(){
    scanf("%d%d",&n,&m);
    for(int i=1;i<=m;i++){
        int u,v;    scanf("%d%d",&u,&v);
        u++;    v++;
        g[u].push_back(v);
        g[v].push_back(u);
        
    }
    scanf("%d",&k);
    memset(p,0,sizeof p);
    for(int i=1;i<=k;i++){
        scanf("%d",&a[i]);
        a[i]++;
        p[a[i]]=true;
        
    }
    
    for(int i=1;i<=n;i++){
        f[i]=i;    sz[i]=1;
    }
    cnt=0;
    for(int i=1;i<=n;i++)
    if(!p[i]&&f[i]==i){
        cnt++;
        s[hd=tl=1]=i;
        while(hd<=tl){
            int u=s[hd++];
            for(UI j=0;j<g[u].size();j++){
                int v=g[u][j];
                if(f[v]==i||p[v])    continue;
                
                f[v]=i;
                sz[v]=sz[u]+1;
                sz[i]=max(sz[i],sz[v]);
                s[++tl]=v;
                
            }
            
        }
        
    }
        
    ans[k]=cnt;
    for(int i=k;i>=1;i--){
        cnt++;
        for(UI j=0;j<g[a[i]].size();j++)
        if(!p[g[a[i]][j]]){
            int x=find(a[i]),y=find(g[a[i]][j]);
            if(x!=y){
                merge(x,y);    cnt--;
            }
            
        }
        p[a[i]]=false;
        ans[i-1]=cnt;
        
    }
    
    for(int i=0;i<=k;i++)
        printf("%d
",ans[i]);

    return 0;

}
View Code

 

以上是关于JSOI2008星球大战的主要内容,如果未能解决你的问题,请参考以下文章

BZOJ 1015: [JSOI2008]星球大战starwar

[JSOI2008]星球大战starwar

[Luogu 1197] JSOI2008 星球大战

bzoj 1015: [JSOI2008]星球大战starwar

bzoj1015 [JSOI2008]星球大战starwar

P1197 [JSOI2008]星球大战