1. 程式人生 > >【洛谷P1197】[JSOI2008]星球大戰

【洛谷P1197】[JSOI2008]星球大戰

style 星球大戰 輸出 多個 大戰 urn 編號 ace include

題目描述

很久以前,在一個遙遠的星系,一個黑暗的帝國靠著它的超級武器統治者整個星系。某一天,憑著一個偶然的機遇,一支反抗軍摧毀了帝國的超級武器,並攻下了星系中幾乎所有的星球。這些星球通過特殊的以太隧道互相直接或間接地連接。

但好景不長,很快帝國又重新造出了他的超級武器。憑借這超級武器的力量,帝國開始有計劃地摧毀反抗軍占領的星球。由於星球的不斷被摧毀,兩個星球之間的通訊通道也開始不可靠起來。現在,反抗軍首領交給你一個任務:給出原來兩個星球之間的以太隧道連通情況以及帝國打擊的星球順序,以盡量快的速度求出每一次打擊之後反抗軍占據的星球的連通快的個數。(如果兩個星球可以通過現存的以太通道直接或間接地連通,則這兩個星球在同一個連通塊中)。

輸入輸出格式

輸入格式:

輸入文件第一行包含兩個整數,N (1 <= N <= 2M) 和M (1 <= M <= 200,000),分別表示星球的數目和以太隧道的數目。星球用0~N-1的整數編號。

接下來的M行,每行包括兩個整數X, Y,其中(0<=X<>Y<N),表示星球X和星球Y之間有以太隧道。註意所有的以太隧道都是雙向的。

接下來一行是一個整數K,表示帝國計劃打擊的星球個數。

接下來的K行每行一個整數X,滿足0<=X<N,表示帝國計劃打擊的星球編號。帝國總是按輸入的順序依次摧毀星球的。

輸出格式:

輸出文件的第一行是開始時星球的連通塊個數。

接下來的K行,每行一個整數,表示經過該次打擊後現存星球的連通塊個數。

輸入輸出樣例

輸入樣例#1:
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
輸出樣例#1:
1
1
1
2
3
3

說明

[JSOI2008]

分析

題目大意:從一個無向圖中去掉一些點,求連通塊的個數。

知識點:貪心,並查集。

這道題有多個查詢,我們可以考慮用離線做法。這就變成了在去掉k個點的基礎下,每加一個點連通塊的個數。

去掉k個點之後連通塊的個數最多是n-k個(每個點互不相通)。我們考慮連通塊減少的條件:

如果加入這個點這個點只屬於某一個連通塊,那麽連通塊的個數不會減少。

如果加入這個點這個點屬於兩個連通塊,連通塊的個數-1,然後再將這兩個連通塊合並。

這樣的話逆序做一遍就行了。

代碼

#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
using namespace std;
const int maxn=400000+5;
inline int read()
{
    int x=0,f=1; char ch=getchar();
    while(ch<0||ch>9){if(ch==-)f=-1; ch=getchar();}
    while(ch>=0&&ch<=9){x=x*10+ch-0; ch=getchar();}
    return x*f;
}
int n,m,k,num,tot;
int head[maxn],father[maxn],a[maxn],b[maxn];
bool vis[maxn];
struct node
{
    int from,next,to;
}e[maxn];
inline void add(int from,int to)
{
    e[++num].next=head[from];
    e[num].from=from;
    e[num].to=to;
    head[from]=num;
}
int find(int x)
{
    if(x!=father[x]) father[x]=find(father[x]);
        return father[x];
}
int main()
{
    n=read();m=read();
    for(int i=0;i<n;i++) father[i]=i;
    for(int i=1;i<=m;i++)
    {
        int x,y;
        x=read();y=read();
        add(x,y);add(y,x);
    }
    k=read(); tot=n-k;
    for(int i=1;i<=k;i++) 
    { a[i]=read(); vis[a[i]]=1;}
    for(int i=1;i<=num;i=i+2)//因為建的是雙向邊
    {
        if(!vis[e[i].from]&&!vis[e[i].to])
        {
            int r1=find(e[i].from);
            int r2=find(e[i].to);
            if(r1!=r2)
            {
                tot--;
                father[r1]=r2;
            }
        }
    }
    b[k+1]=tot;
    for(int j=k;j>=1;j--)
    {
        int u=a[j]; vis[u]=0; tot++;
        int r1=find(u);//這兒是一個小技巧,不更改u的父親
        for(int i=head[u];i;i=e[i].next)
        if(!vis[e[i].to])
        {
            int r2=find(e[i].to);
            if(r1!=r2){tot--; father[r2]=r1;}
        }//合並的時候把r2合並到r1上,這樣u的父親就不會變,減少計算
        b[j]=tot;
    }
    for(int i=1;i<=k+1;i++)
    printf("%d\n",b[i]);
    return 0;
}

【洛谷P1197】[JSOI2008]星球大戰