生活随笔
收集整理的這篇文章主要介紹了
洛谷1197星球大战
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
題目描述
很久以前,在一個遙遠的星系,一個黑暗的帝國靠著它的超級武器統治者整個星系。某一天,憑著一個偶然的機遇,一支反抗軍摧毀了帝國的超級武器,并攻下了星系中幾乎所有的星球。這些星球通過特殊的以太隧道互相直接或間接地連接。
但好景不長,很快帝國又重新造出了他的超級武器。憑借這超級武器的力量,帝國開始有計劃地摧毀反抗軍占領的星球。由于星球的不斷被摧毀,兩個星球之間的通訊通道也開始不可靠起來。現在,反抗軍首領交給你一個任務:給出原來兩個星球之間的以太隧道連通情況以及帝國打擊的星球順序,以盡量快的速度求出每一次打擊之后反抗軍占據的星球的連通快的個數。(如果兩個星球可以通過現存的以太通道直接或間接地連通,則這兩個星球在同一個連通塊中)。
輸入輸出格式
輸入格式:
輸入文件第一行包含兩個整數,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]
由于題目沒有要求在線, 考慮離線做法.
使用并査集. 逆向維護加點.
代碼如下. 目前是WA的, 但仍然可以用作參考. 歡迎查錯(最主要是這題好難調試的說). A了以后會回來改.
#include<iostream>
#include<string.h>
#include<stdio.h>
using namespace std;
const int maxN = (
int)
1e5;
const int maxM = (
int)
2e5;
const int maxK = maxN;
int n;
int head[maxN];
int size;
struct edge
{
int v, next;
}G[maxM <<
1];
void addEdge(
int u,
int v)
{G[size].v = v;G[size].next = head[u];head[u] = size ++;
}
int Q[maxK];
int tag[maxN];
int pre[maxN];
void getPre(
int u)
{
if(u == pre[u])
return;getPre(pre[u]);pre[u] = pre[pre[u]];
}
int rec[maxN];
int ans[maxK +
1];
void count(
int x)
{
memset(rec,
0,
sizeof(rec));
int cnt =
0;
for(
int i =
0; i < n; i ++)
if(! tag[i]){getPre(i);
if(! rec[pre[i]])cnt ++;rec[pre[i]] ++;}ans[x] = cnt;
}
int main()
{
int m;
cin >> n >> m;size =
0;
memset(head, -
1,
sizeof(head));
for(
int i =
0; i < m; i ++){
int u, v;
cin >> u >> v;addEdge(u, v);addEdge(v, u); }
int K;
cin >> K;
memset(tag,
0,
sizeof(tag));
for(
int i =
0; i < K; i ++)
cin >> Q[i], tag[Q[i]] =
1;
for(
int i =
0; i < n; i ++)pre[i] = i;
for(
int i =
0; i < n; i ++)
if(! tag[i])
for(
int j = head[i]; j != -
1; j = G[j].next){
int v = G[j].v;
if(tag[v])
continue;getPre(i);getPre(v);pre[pre[v]] = pre[i];}count(K);
for(
int i = K -
1; i >=
0; i --){
int u = Q[i];
for(
int j = head[u]; j != -
1; j = G[j].next){
int v = G[j].v;
if(tag[v])
continue;getPre(u);getPre(v);pre[pre[v]] = pre[u];}count(i);tag[u] =
0;}
for(
int i =
0; i <= K; i ++)
printf(
"%d\n", ans[i]);
}
轉載于:https://www.cnblogs.com/ZeonfaiHo/p/6402874.html
總結
以上是生活随笔為你收集整理的洛谷1197星球大战的全部內容,希望文章能夠幫你解決所遇到的問題。
如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。