標籤:ret 通訊 pre 輸入 直接 編號 它的 main str
題目描述
很久以前,在一個遙遠的星系,一個黑暗的帝國靠著它的超級武器統治者整個星系。某一天,憑著一個偶然的機遇,一支反抗軍摧毀了帝國的超級武器,並攻下了星系中幾乎所有的星球。這些星球通過特殊的以太隧道互相直接或間接地串連。
但好景不長,很快帝國又重新造出了他的超級武器。憑藉這超級武器的力量,帝國開始有計劃地摧毀反抗軍佔領的星球。由於星球的不斷被摧毀,兩個星球之間的通訊通道也開始不可靠起來。現在,反抗軍首領交給你一個任務:給出原來兩個星球之間的以太隧道連通情況以及帝國打擊的星球順序,以盡量快的速度求出每一次打擊之後反抗軍佔據的星球的連通快的個數。(如果兩個星球可以通過現存的以太通道直接或間接地連通,則這兩個星球在同一個連通塊中)。
輸入輸出格式
輸入格式:
輸入檔案第一行包含兩個整數,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 130 11 66 55 00 61 22 33 44 57 17 27 63 6516357
輸出範例#1:
111233
說明
[JSOI2008]
關鍵詞:逆向,並查集,離線
//Gang #include<iostream>#include<cstring>#include<algorithm>#include<cstdio>#include<cstdlib>#include<queue>#include<cmath>#define FOR(x,y,z) for(int x=y;x<=z;x++)#define REP(x,y,z) for(int x=y;x>=z;x--)#define ll long longusing namespace std;int fa[500005],hd[500005];int t[500005],vis[500005];int cnt;int n,m,k;int out[500005];int ans;int a,b;struct node{ int to,next;}e[500005];int find(int x){ int p,temp; p=x; while(x!=fa[x]) { x=fa[x]; } while(p!=x) { temp=fa[p]; fa[p]=x; p=temp; } return x;}void add(int x,int y){ ++cnt; e[cnt].next=hd[x]; e[cnt].to=y; hd[x]=cnt;}int main(){scanf("%d%d",&n,&m);FOR(i,0,n-1)fa[i]=i; //從0開始 ,很重要!!!FOR(i,1,m){ scanf("%d%d",&a,&b); add(a,b),add(b,a);//無向圖}scanf("%d",&k);FOR(i,1,k){ scanf("%d",&t[i]); vis[t[i]]=true;}ans=n-k;FOR(i,0,n-1) //找出不會被轟炸的點並連起來{ if(!vis[i]) { for(int j=hd[i];j;j=e[j].next) { if(!vis[e[j].to]) { int dx=find(i),dy=find(e[j].to); if(dx!=dy) fa[dy]=dx,ans--; } } }}out[k+1]=ans;REP(i,k,1){ int c=t[i]; vis[c]=false; ans++;//因為新加了一個點,所以連通塊的個數要加一 for(int j=hd[c];j;j=e[j].next) { if(!vis[e[j].to]) { int dx=find(c),dy=find(e[j].to); if(dx!=dy) fa[dy]=dx,ans--; } } out[i]=ans;}FOR(i,1,k+1)printf("%d\n",out[i]); return 0;}
P1197 [JSOI2008]星球大戰