pat甲级1107 并查集

并查集在findFather()函数中进行压缩路径,陷阱是这里只压缩该结点以上到根的路径,其以下的路径不压缩,这里不搞清楚会有三个测试点过不去

#include <cstdio>
#include <vector>
#include <algorithm>

using namespace std;

int N;
vector<int> hobby[1001];
int father[1001];
int cluster[1001] = {};

int findFather(int a);
void unionElem(int a, int b);
bool cmp(int a, int b);

int main(){
    scanf("%d", &N);
    for(int i=1; i<=N; i++) father[i] = i; //并查集初始化
    for(int i=1; i<=N; i++){
        int ct, h;
        scanf("%d:", &ct);
        while(ct--){
            scanf("%d", &h);
            hobby[h].push_back(i);
        }
    }
    for(int i=1; i<=1000; i++){
        int n = hobby[i].size();
        for(int j=1; j<n; j++) unionElem(hobby[i][0], hobby[i][j]);
    }
    for(int i=1; i<=N; i++){
        cluster[findFather(i)]++; //必须用findFather(i),不能用father[i]
    }
    sort(cluster, cluster+1001, cmp);
    int clu = 0;
    while(cluster[clu]>0) clu++;
    printf("%d\n", clu);
    for(int i=0; i<clu; i++){
        printf("%d", cluster[i]);
        if(i<clu-1) printf(" ");
    }

    return 0;
}

int findFather(int a){
    int x=a;
    while(father[a]!=a) a = father[a];
    while(father[x]!=x){
        //压缩该结点以上的路径
        int t=x;
        x = father[x];
        father[t] = a;
    }
    return a;
}

void unionElem(int a, int b){
    int fa = findFather(a);
    int fb = findFather(b);
    if(fa!=fb) father[fb] = fa;
    return;
}

bool cmp(int a, int b){
    return a>b;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值