zoukankan      html  css  js  c++  java
  • 【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
  • 相关阅读:
    ARC109C Large RPS Tournament 机智
    ABC186F Rook on Grid 树状数组
    二分查找
    CF1445D. Divide and Sum 组合数
    APP测试方法分享
    面试常见问题
    接口测试基础知识
    接口测试一
    web端测试
    Jmeter简介
  • 原文地址:https://www.cnblogs.com/Hansue/p/12913231.html
Copyright © 2011-2022 走看看