并查集的两种实现(按秩合并+路径压缩)

Posted 2018zxy

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了并查集的两种实现(按秩合并+路径压缩)相关的知识,希望对你有一定的参考价值。

并查集:就是有求并集,查找元素属于哪个集合的功能。

 

1、路径压缩:使X到根上的每一个节点的父节点都变为根节点。

查询:

void Find(int x)
{
    if(a[x]==0) return x;
    else return a[x]=Find(a[x]);
}

 

合并:

void Merge(int x,int y)
{
    int t1=Find(x),t2=Find(y);
    if(t1!=t2) a[t1]=t2;
}

 

2、按秩合并:使较浅的树成为较深的树的子树。

查询:

void Find(int x)
{
    if(a[x]==0) return x;
    else return a[x]=Find(a[x]);
}

 

合并:

void Merge(int x,int y)
{
    int t1=Find(x),t2=Find(y);
    if(t1!=t2){
        if(rank[t1]<=rank[t2]) a[t1]=t2,rank[t2]=max(rank[t2],rank[t1]+1);
        else a[t2]=t1,rank[t1]=max(rank[t1],rank[t2]+1);
    }
}

 

例题:hdu1232

解法一:路径压缩

技术图片
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int maxn = 1200;
int fa[maxn];
int f(int x)
{
    if(fa[x]==0) return x;
    else return fa[x]=f(fa[x]);
}
int main(void)
{
    int n,m,t1,t2,i,x,y;
    while(~scanf("%d",&n)&&n){
        scanf("%d",&m);
        memset(fa,0,sizeof(fa));
        while(m--){
            scanf("%d%d",&x,&y);
            t1=f(x),t2=f(y);
            if(t1!=t2) fa[t1]=t2;
        }
        int cnt=0;
        for(i=1;i<=n;i++){
            if(fa[i]==0) cnt++;
        }
        printf("%d
",cnt-1);
    }
    return 0;
}
View Code

解法二:按秩求和

技术图片
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int maxn = 1200;
int fa[maxn],rk[maxn];
int f(int x){
    if(fa[x]==0) return x;
    else return fa[x]=f(fa[x]);
}
int MAX(int x,int y)
{
    return x>y?x:y;
}
int main(void)
{
    int n,m,x,y,t1,t2,i;
    while(~scanf("%d",&n)&&n){
        scanf("%d",&m);
        memset(fa,0,sizeof(fa));
        memset(rk,0,sizeof(rk));
        while(m--){
            scanf("%d%d",&x,&y);
            t1=f(x),t2=f(y);
            if(t1!=t2){
                if(rk[t1]<=rk[t2]) fa[t1]=t2,rk[t1]=MAX(rk[t1],rk[t2]+1);
                else fa[t2]=t1,rk[t2]=MAX(rk[t2],rk[t1]+1);
            }
        }
        int cnt=0;
        for(i=1;i<=n;i++)
        if(fa[i]==0) cnt++;
        printf("%d
",cnt-1);
    }
    return 0;
}
View Code

 

以上是关于并查集的两种实现(按秩合并+路径压缩)的主要内容,如果未能解决你的问题,请参考以下文章

并查集

关于并查集的一切全在这里了

bzoj3674 可持久化并查集加强版

可持久化并查集

并查集

bzoj4668: 冷战 并查集按秩合并