其他分享
首页 > 其他分享> > 红色警报 并查集

红色警报 并查集

作者:互联网

#include <bits/stdc++.h>
using namespace std;
const int maxn=501,maxsize=5001;
int father[maxn];
bool visited[maxn]={false};//表示该城市是否被攻占
struct Edge
{
       int u;
       int v;
};
struct Edge edge[maxsize];
int findfather(int x)
{
     if(x==father[x])
              return x;
     else
     {
        int F=findfather(father[x]);//递归找父亲
        father[x]=F;//由低到高 分别把f赋给每一个人
        return F;

     }
}
void Union(int a,int b)
{
    int fa=findfather(a);
    int fb=findfather(b);
    if(fa!=fb)
        father[fa]=fb;

}
int main()
{
    int n,m,k,num1=0,num2=0;
    cin>>n>>m;
    for(int i=0;i<n;++i)
       father[i]=i;
    for(int i=0;i<m;++i)
    {
       cin>>edge[i].u>>edge[i].v;
       Union(edge[i].u,edge[i].v);

    }
    for(int i=0;i<n;++i)
    {
      if(father[i]==i)
      {
          num1++;
      }
    }
    cin>>k;
    int cnt=k;
    while(k--)
    {
       num2=0;
       int city;
       cin>>city;
       visited[city]=true;//排除这个点重新建立新的并查集
           for(int i=0;i<n;++i)
              father[i]=i;
       for(int i=0;i<m;++i)//重新建立并查集
       {
           if(visited[edge[i].u]==true||visited[edge[i].v]==true)//只要两个连通的城市有一个已经被干掉 就跳过
           {
              continue;
           }
           Union(edge[i].u,edge[i].v);
       }
      for(int i=0;i<n;++i)//重新数一下有多少个连通块 去掉之前攻占的
       {
           if(visited[i]==false&&father[i]==i)//由于还是会访问之前被攻占城市所以要去除
                     num2++;

       }
       if(num2==num1||num2==num1-1)//连通块不变或者说连通块少一个没有影响
       {
              printf("City %d is lost.\n",city);
              num1=num2;
       }
       else if(num2>num1)
       {
            printf("Red Alert: City %d is lost!\n",city);
            num1=num2;
       }
    }
       if(cnt==n)
       {
          printf("Game Over.\n");
       }



    return 0;
}

 

标签:city,num2,int,查集,father,红色警报,edge,findfather
来源: https://blog.csdn.net/qq_40795475/article/details/88699173