zoukankan      html  css  js  c++  java
  • 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;
    }
    
    
    
  • 相关阅读:
    linux安装jenkins
    如何将接口进行限流
    java线程池思想
    一次缓存评估过程
    docker
    linux基本操作
    【安卓】App自动化环境搭建
    sheill之文本处理工具
    Liunx vim编辑器
    Liunx 远程
  • 原文地址:https://www.cnblogs.com/keyboarder-zsq/p/5934468.html
Copyright © 2011-2022 走看看