并查集【p1197】[JSOI2008]星球大战

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

第一行是开始时星球的连通块个数。接下来的 \(K\) 行,每行一个整数,表示经过该次打击后现存星球的连通块个数。

维护联通块个数? 考虑并查集

但是正序不好做,因此我们考虑倒叙.(正难则反思想)

我们加入这些被破坏的点.

然后再去判断联通块个数,注意新加入一个点的话,如果没有联通,联通块个数要\(++\),如果还能和其他构成联通块,要\(--\)

代码

#include
#include
#include
#define N 500008
#define R register
using namespace std;

inline void in(int &x)
{
    int f=1;x=0;char s=getchar();
    while(!isdigit(s)){if(s=='-')f=-1;s=getchar();}
    while(isdigit(s)){x=x*10+s-'0';s=getchar();}
    x*=f;
}

int n,m,f[N],k;
int head[N],tot,ques[N],ans[N],cnt;
struct cod{int u,v;}edge[N<<2];
inline void add(R int x,R int y)
{
    edge[++tot].u=head[x];
    edge[tot].v=y;
    head[x]=tot;
}
bool bom[N];
int find(R int x){return f[x]==x?x:f[x]=find(f[x]);}
int main()
{
    in(n),in(m);
    for(R int i=1,x,y;i<=m;i++)
    {
        in(x),in(y);
        x++,y++;
        add(x,y),add(y,x);
    }
    in(k);
    for(R int i=1,x;i<=k;i++)
    {
        in(x);x++;
        bom[x]=true;
        ques[i]=x;
    }
    for(R int i=1;i<=n;i++)f[i]=i;
    cnt=n-k;
    for(R int x=1;x<=n;x++)
    {
        if(bom[x])continue;
        for(R int i=head[x];i;i=edge[i].u)
        {
            if(bom[edge[i].v])continue;
            R int fu=find(x),fv=find(edge[i].v);
            if(fu==fv)continue;
            f[fu]=fv;cnt--;
        }
    }
    ans[k+1]=cnt;
    for(R int j=k;j>=1;j--)
    {
        R int x=ques[j];
        bom[x]=false;cnt++;
        for(R int i=head[x];i;i=edge[i].u)
        {
            if(bom[edge[i].v])continue;
            R int fu=find(x),fv=find(edge[i].v);
            if(fu==fv)continue;
            f[fu]=fv;cnt--;
        }
        ans[j]=cnt;
    }
    for(R int i=1;i<=k+1;i++)
        printf("%d\n",ans[i]);
}

转载于:https://www.cnblogs.com/-guz/p/9867236.html

你可能感兴趣的:(并查集【p1197】[JSOI2008]星球大战)