zoukankan      html  css  js  c++  java
  • BZOJ 1015: [JSOI2008]星球大战starwar(并查集求连通块+离线处理)

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

    题意:

    思路:
    好题啊!!!

    这道题目需要离线处理,先把所有要删的点给保存下来,然后逆序加点,这样就把原来的删点变为了加点,加点的话计算连通块就方便的多,具体参见代码。

      1 #include<iostream>
      2 #include<algorithm>
      3 #include<cstring>
      4 #include<cstdio>
      5 #include<sstream>
      6 #include<vector>
      7 #include<stack>
      8 #include<queue>
      9 #include<cmath>
     10 #include<map>
     11 #include<set>
     12 using namespace std;
     13 typedef long long ll;
     14 typedef pair<int,ll> pll;
     15 const int INF = 0x3f3f3f3f;
     16 const int maxn=200000+5;
     17 
     18 int n, m;
     19 int k;
     20 int tot;
     21 int cnt;
     22 int q[2*maxn];
     23 int p[2*maxn];
     24 int vis[2*maxn];
     25 int del[2*maxn];
     26 int ans[2*maxn];
     27 int head[2*maxn];
     28 
     29 struct node
     30 {
     31     int v,next;
     32 }e[2*maxn];
     33 
     34 void addEdge(int u,int v)
     35 {
     36     e[tot].v=v;
     37     e[tot].next=head[u];
     38     head[u]=tot++;
     39 }
     40 
     41 int Find(int x)
     42 {
     43     return x==p[x]?x:p[x]=Find(p[x]);
     44 }
     45 
     46 void update(int u)
     47 {
     48     for(int i=head[u];i!=-1;i=e[i].next)
     49     {
     50         int v=e[i].v;
     51         if(vis[v])  //如果v之前已经访问过了并且u和v不在同一连通块中,那么合并并且减少一个连通块
     52         {
     53             int x=Find(u);
     54             int y=Find(v);
     55             if(x!=y)  {p[x]=y;cnt--;}
     56         }
     57     }
     58 }
     59 
     60 int main()
     61 {
     62     //freopen("in.txt","r",stdin);
     63     while(~scanf("%d%d",&n,&m))
     64     {
     65         tot=0;
     66         memset(head,-1,sizeof(head));
     67         memset(del,0,sizeof(del));
     68         memset(vis,0,sizeof(vis));
     69         for(int i=0;i<n;i++)  p[i]=i;
     70         while(m--)
     71         {
     72             int u,v;
     73             scanf("%d%d",&u,&v);
     74             addEdge(u,v);
     75             addEdge(v,u);
     76         }
     77 
     78         scanf("%d",&k);
     79         for(int i=1;i<=k;i++)  {scanf("%d",&q[i]);del[q[i]]=1;}
     80 
     81         cnt=0;
     82         for(int i=0;i<n;i++)
     83         {
     84             if(!del[i])
     85             {
     86                 cnt++; //先把它单独作为一个连通块
     87                 update(i);
     88                 vis[i]=1;
     89             }
     90         }
     91 
     92         ans[k+1]=cnt;
     93         for(int i=k;i>=1;i--)  //加点
     94         {
     95             cnt++;
     96             update(q[i]);
     97             vis[q[i]]=1;
     98             ans[i]=cnt;
     99         }
    100         for(int i=1;i<=k+1;i++)  printf("%d
    ",ans[i]);
    101     }
    102     return 0;
    103 }
  • 相关阅读:
    c++中的内存管理【转载】
    c++中dynamic_cast、static_cast、reinterpret_cast和const_cast作用
    c++中的顶层const和底层const
    c++赋值操作符需要确保自我赋值的安全性问题
    二分法查找
    Servlet基础总结
    java 正则表达式:有丶东西
    HTTP协议初步认识
    Java synchronized到底锁住的是什么?
    ECMA Script 6新特性之解构赋值
  • 原文地址:https://www.cnblogs.com/zyb993963526/p/7337674.html
Copyright © 2011-2022 走看看