PTA 朋友圈【并查集的合并问题】

这里写图片描述

一开始,考虑的是每次就是把第一个作为祖先,这样很明显是错误的,比如
7 4
3 1 2 3
2 4 2
3 5 6 7
1 6
所以这正是更好地体现对于集合的代表。只有把所有的元素合并一下,然后选一个作为代表代表集合,这有点感觉强连通算法。

所以后来的比较好的方法,就是每个都并一下,选一个作为代表,因为并的操作是要find的,而find是要找那个boss的,所以很不错诶;

#include<iostream>
#include<cstdio>
#include<math.h>
#include<queue>
#include<map>
#include<stdlib.h>
#include<string>
#include<string.h>
#include<algorithm>
using namespace std;
typedef long long LL;
#define PI acos(-1.0)

const int N=1e4+10;

int pre[N*3];
int num[N*3];

int Find(int x)
{
    int r=x;
    while(pre[r]!=r)
        r=pre[r];
    int i=x,j;
    while(pre[i]!=r)
    {
        j=pre[i];
        pre[i]=r;
        i=j;
    }
    return r;
}

void Union(int a,int b)
{
    int aa=Find(a);
    int bb=Find(b);
    if(aa!=bb)
        pre[aa]=bb;
}

int main()
{
    int n,m;
    cin>>n>>m;
    for(int i=1;i<=n;i++)
        pre[i]=i;
    for(int i=1;i<=m;i++)
    {
        int a,b,x,k;
        scanf("%d",&a);
        scanf("%d",&b);
        for(int j=1;j<a;j++)
        {
            scanf("%d",&x);
            Union(b,x);
        }
    }
    memset(num,0,sizeof(num));
    for(int i=1;i<=n;i++)
    {
        int x=Find(i);
            num[x]++;
    }
    sort(num+1,num+n+1);
    printf("%d
",num[n]);
    return 0;
}


原文地址:https://www.cnblogs.com/keyboarder-zsq/p/5934468.html