zoukankan      html  css  js  c++  java
  • BZOJ_1015_星球大战_[JSOI2008]_(并查集)

    描述


     

    http://www.lydsy.com/JudgeOnline/problem.php?id=1015

    n 个点,被 m 条边相连.进行k次删点操作,问第一次操作前和每次操作后的集合数(直接或间接连在一起的点属于同一个集合).

    1015: [JSOI2008]星球大战starwar

    Time Limit: 3 Sec  Memory Limit: 162 MB
    Submit: 4696  Solved: 2114
    [Submit][Status][Discuss]

    Description

      很久以前,在一个遥远的星系,一个黑暗的帝国靠着它的超级武器统治者整个星系。某一天,凭着一个偶然的
    机遇,一支反抗军摧毁了帝国的超级武器,并攻下了星系中几乎所有的星球。这些星球通过特殊的以太隧道互相直
    接或间接地连接。 但好景不长,很快帝国又重新造出了他的超级武器。凭借这超级武器的力量,帝国开始有计划
    地摧毁反抗军占领的星球。由于星球的不断被摧毁,两个星球之间的通讯通道也开始不可靠起来。现在,反抗军首
    领交给你一个任务:给出原来两个星球之间的以太隧道连通情况以及帝国打击的星球顺序,以尽量快的速度求出每
    一次打击之后反抗军占据的星球的连通快的个数。(如果两个星球可以通过现存的以太通道直接或间接地连通,则
    这两个星球在同一个连通块中)。

    Input

      输入文件第一行包含两个整数,N (1  < =  N  < =  2M) 和M (1  < =  M  < =  200,000),分别表示星球的
    数目和以太隧道的数目。星球用 0 ~ N-1的整数编号。接下来的M行,每行包括两个整数X, Y,其中(0 < = X <>
    Y 表示星球x和星球y之间有“以太”隧道,可以直接通讯。接下来的一行为一个整数k,表示将遭受攻击的星球的
    数目。接下来的k行,每行有一个整数,按照顺序列出了帝国军的攻击目标。这k个数互不相同,且都在0到n-1的范
    围内。

    Output

      输出文件的第一行是开始时星球的连通块个数。接下来的N行,每行一个整数,表示经过该次打击后现存星球
    的连通块个数。

    Sample Input

    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

    Sample Output

    1
    1
    1
    2
    3
    3

    HINT

    Source

    分析


    离线处理,先把所有点都删掉,建立并查集,统计集合数,这就是最后一个答案,然后加上最后一个被删掉的点,这就是倒数第二个答案,以此类推,直到将所有点都加回去,就是进行删除操作前的答案.

    怎么统计集合数呢?每来一个点,集合数 tot ++ ,如果发现有可以合并的集合,那就合并集合,并 tot -- .

     1 #include<cstdio>
     2 #include<algorithm>
     3 #include<vector>
     4 #define read(a) a=getnum()
     5 using namespace std;
     6 
     7 const int maxm=200005,maxn=400005;
     8 int n,m,k;
     9 int f[maxn],q[maxn],ans[maxn];
    10 bool mark[maxn];
    11 vector <int> g[maxn];
    12 
    13 inline int getnum(){int ret=0,k=1;char c;for(c=getchar();c<'0'||c>'9';c=getchar()) if(c=='-') k=-1;for(;c>='0'&&c<='9';c=getchar()) ret=ret*10+c-'0'; return ret*k;}
    14 
    15 int find(int x) { return x==f[x]?x:f[x]=find(f[x]); }
    16 
    17 void unite(int x,int y)
    18 {
    19     int fx=find(x),fy=find(y);
    20     if(fx!=fy) f[fx]=fy;
    21 }
    22 
    23 void update(int x,int &tot)
    24 {
    25     mark[x]=false;
    26     for(int i=0;i<g[x].size();i++)
    27     {
    28         int y=g[x][i];
    29         if(mark[y]) continue;
    30         if(find(x)!=find(y))
    31         {
    32             unite(x,y);
    33             tot--;
    34         }
    35     }
    36 }
    37     
    38 
    39 void solve()
    40 {
    41     for(int i=1;i<=n;i++) f[i]=i;
    42     int tot=0; 
    43     for(int i=1;i<=n;i++)
    44     {
    45         if(mark[i]) continue;
    46         tot++;
    47         update(i,tot);
    48     }
    49     ans[k]=tot;
    50     for(int i=k;i>=1;i--)
    51     {
    52         tot++;
    53         update(q[i],tot);
    54         ans[i-1]=tot;
    55     }
    56     for(int i=0;i<=k;i++) printf("%d
    ",ans[i]);
    57 }
    58 
    59 void init()
    60 {
    61     read(n); read(m);
    62     for(int i=1;i<=m;i++)
    63     {
    64         int a,b;
    65         read(a); read(b);
    66         a++; b++;
    67         g[a].push_back(b);
    68         g[b].push_back(a);
    69     }
    70     read(k);
    71     for(int i=1;i<=k;i++)
    72     {
    73         read(q[i]);
    74         q[i]++;
    75         mark[q[i]]=true;
    76     }
    77 }
    78 
    79 int main()
    80 {
    81 #ifndef ONLINE_JUDGE
    82     freopen("star.in","r",stdin);
    83     freopen("star.out","w",stdout);
    84 #endif
    85     init();
    86     solve();
    87 #ifndef ONLINE_JUDGE
    88     fclose(stdin);
    89     fclose(stdout);
    90     system("star.out");
    91 #endif
    92     return 0;
    93 }
    View Code
  • 相关阅读:
    JSP中文乱码的一点研究
    Eclipse 导入外部项目无法识别为web项目并且无法在部署到tomcat下
    DOS命令学习
    Oracle学习笔记(一)
    登录时候的图片验证码的实现
    DIY随机数产生类
    使用VS2010建模
    Random快速连续产生不相同随机数
    魔兽争霸3冰封王座英雄以及兵种语言
    取两个地图坐标之间的距离
  • 原文地址:https://www.cnblogs.com/Sunnie69/p/5433034.html
Copyright © 2011-2022 走看看