解题思路:并查集裸题,我需要用并查集查找出帮派的个数,最后只需要修帮派个数减一条道路就可以将所有帮派连通起来但是数据量太大需要用scanf
下面附上ac代码
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <string>
#include <queue>
#include <set>
#include <map>
#include <unordered_map>
#define dbg(a) cout<<#a<<" : "<<a<<endl;
using namespace std;
typedef long long ll;
const int MAX=1e7+5;
int fa[MAX];
int num[MAX];
int rnk[MAX];
unordered_map<ll,ll> Fa,m;
void init(int n)
{
for(int i=1;i<=n;i++)
{
fa[i]=i;
rnk[i]=1;
}
}
int find(int x)
{
return x==fa[x]?x:(fa[x]=find(fa[x]));
}
void merge(int i,int j)
{
int x=find(i);
int y=find(j);
if(rnk[x]<=rnk[y])
{
fa[x]=y;
}
else{
fa[y]=x;
}
if(rnk[x]==rnk[y] && x!=y)
{
rnk[y]++;
}
}
int main()
{
//std::ios::sync_with_stdio(false);
//cin.tie(0),cout.tie(0);
int n,m;
while(scanf("%d",&n)!=EOF && n)
{
cin>>m;
init(n);
int x,y;
for(int i=1;i<=m;i++)
{
scanf("%d%d",&x,&y);
merge(x,y);
}
ll cnt=0;
for(int i=1;i<=n;i++)
{
if(fa[i]==i)
cnt++;
}
printf("%d\n",cnt-1);
}
return 0;
}