1. 程式人生 > >luoguP1197 [JSOI2008]星球大戰 x

luoguP1197 [JSOI2008]星球大戰 x

tro %d bsp src 出了 ios span bbb play

P1197 [JSOI2008]星球大戰

題目描述

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

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

輸入輸出格式

輸入格式:

輸入文件第一行包含兩個整數,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]

思路:

  聯通塊+並查集的逆用

  一開始假設所有的星球全部摧毀,然後進行離線處理(聽著好高端的樣子!)

坑點:

  一定要記得初始化head數組的值!!!因為會用到0號元素

上代碼:

技術分享
#include <iostream>
#include 
<cstdio> #include <cstring> #include <algorithm> using namespace std; const int M = 4e5 + 5; int heads[M]; int n,m,k; int now=0,x,y; int dad[M],hit[M],answer[M]; bool v[M]; struct A{ int next,to; }h[M]; void add(int pre,int to) { h[now].to=to; h[now].next=heads[pre]; heads[pre]=now++; } int getdad(int Ms) { return dad[Ms] == Ms ? Ms : dad[Ms]=getdad(dad[Ms]); } void unions(int dads,int sons) { dad[getdad(dads)]=getdad(sons); } int main() { scanf("%d%d",&n,&m); for(int i=0;i<n;i++) dad[i]=i,heads[i]=-1; for(int i=0;i<m;i++) { scanf("%d%d",&x,&y); add(x,y),add(y,x); } scanf("%d",&k); for(int i=0;i<k;i++) { scanf("%d",&hit[i]); v[hit[i]]=true; ///因為是逆序,表示當前星球已經被打擊 } for(int i=0;i<n;i++) { if(!v[i]) { for(int j=heads[i];j!=-1;j=h[j].next) { int vc=h[j].to; if(!v[vc]) unions(i,vc); } } } for(int i=0;i<n;i++) if(!v[i] && getdad(i) == i) answer[k]++;///尋找最後一張圖所剩余的連通塊 for(int i=k-1;i>=0;i--)///因為存儲的時候是正序存儲,進行計算的時候應該逆序操作,所以顛倒順序 { int x=0; ///x記錄的是與當前被打擊掉的星球相連的並且不被打擊掉的請求 v[hit[i]]=false;///清除標記,因為是逆序. for(int j=heads[hit[i]];j!=-1;j=h[j].next) { if(!v[h[j].to]) { int a=getdad(hit[i]); int b=getdad(h[j].to); if(a!=b) { unions(a,b); x++; } } } answer[i]=answer[i+1]-x+1; } for(int i=0;i<=k;i++) printf("%d\n",answer[i]); return 0; }
View Code

luoguP1197 [JSOI2008]星球大戰 x