標籤:通過 輸入輸出 依次 雙向 print -- jsoi2008 原來 names
題目描述
很久以前,在一個遙遠的星系,一個黑暗的帝國靠著它的超級武器統治者整個星系。某一天,憑著一個偶然的機遇,一支反抗軍摧毀了帝國的超級武器,並攻下了星系中幾乎所有的星球。這些星球通過特殊的以太隧道互相直接或間接地串連。
但好景不長,很快帝國又重新造出了他的超級武器。憑藉這超級武器的力量,帝國開始有計劃地摧毀反抗軍佔領的星球。由於星球的不斷被摧毀,兩個星球之間的通訊通道也開始不可靠起來。現在,反抗軍首領交給你一個任務:給出原來兩個星球之間的以太隧道連通情況以及帝國打擊的星球順序,以盡量快的速度求出每一次打擊之後反抗軍佔據的星球的連通快的個數。(如果兩個星球可以通過現存的以太通道直接或間接地連通,則這兩個星球在同一個連通塊中)。
輸入輸出格式
輸入格式:
輸入檔案第一行包含兩個整數,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]
一道簡單的並查集題,當然你要倒著並。
一個可惡的錯誤,並查集我竟然忘了初始化。
#include<iostream>#include<cstdio>#include<string.h>#include<algorithm>using namespace std;#define LL long long #define maxn 200009int n,m;int h[2*maxn],nex[2*maxn],to[2*maxn],cnt;int k,ord[maxn];int tot,ans[maxn];bool vis[2*maxn];int f[2*maxn];int find(int x){ while(x!=f[x]) x=f[x]=f[f[x]]; return x;}void bing(int x,int y){ int f1=find(x),f2=find(y); if(f1!=f2) f[f1]=f2,tot--; return ;}int main(){ scanf("%d%d",&n,&m); for(int i=1;i<=n;i++) f[i]=i; for(int i=1,x,y;i<=m;i++) { scanf("%d%d",&x,&y); ++x,++y; to[++cnt]=y,nex[cnt]=h[x],h[x]=cnt; to[++cnt]=x,nex[cnt]=h[y],h[y]=cnt; } scanf("%d",&k); for(int i=1;i<=k;i++) scanf("%d",&ord[i]),vis[++ord[i]]=1; tot=n-k;int u,v; for(int i=1;i<=n;i++) if(!vis[i]) for(int j=h[i];j;j=nex[j]) { u=to[j]; if(!vis[u]) bing(i,u); } ans[k]=tot; for(int i=k-1;i>=0;i--) { u=ord[i+1];vis[u]=0;tot++; for(int j=h[u];j;j=nex[j]) { v=to[j]; if(!vis[v]) bing(u,v); } ans[i]=tot; } for(int i=0;i<=k;i++) printf("%d\n",ans[i]); return 0;}
P1197 [JSOI2008]星球大戰