小谈并查集及其算法实现

 

并查集

 

一、算法介绍:

并查集(Union-find Sets)是一种很静止而有用的数据结构,它主要用于处理一些不相交集合的合并问题。

 

并查集的基本操作有两个:

1:合并

union(x, y):把元素 x 和元素 y 所在的集合合并。要求 x 和 y 所在的集合不相交。假设两个集合相交则不合并。

2:查询

find(x):找到元素 x 所在的集合的代表。该操作经常使用于推断两个元素是否位于同一个集合。仅仅要将它们各自的代表比較一下就能够了。

 

并查集类似一个森林,是用树形结构来实现的。所以下面的解说用树形结构来构造模型:

事先声明:

(1)一个集合相应一棵树。

(2)一个集合中的元素相应一个节点。

 

二、算法实现:

 

一、初始化:

我们用n个节点表示n个元素。有一点要特别注意:一个节点。若它的父节点等于它本身,则说明这个节点是根节点。

定义数组 per[], per[x]代表x的父节点。

初始化时我们把per[x] = x,相当于每一个节点都是独立的根节点(每一个根节点都代表一个集合)

 

//n 代表一共同拥有n个元素(n个节点)
for(int i = 1; i <= n; ++i){
	per[i] = i;
}

二、查找:

find(x),查找元素x所在的集合。即查找节点x在哪一棵树上,这里我们知道,在一棵树中,根节点是唯一的。父节点子节点都是相对而言的。要想确定节点x在哪一课树,我们仅仅须要找到x的根节点就能够了。

假设推断节点x 和 节点y在不在同一棵树上,我们仅仅须要找到x 和 y的根节点,若x和y的根节点同样,则x,y在同一颗树上。否则在不用的树上。

代码:

 

int find(int x){    
    int r = x;    
    //父节点等于自身的节点才是根节点。    
    //若 r 节点的父节点不是根节点,一直向上找。    
    if(r != per[x])
        r = per[x];
    return r;
}

 

1.1

如图1.1所看到的:

我们令节点1为根节点。查找节点4在哪一棵树,我们仅仅要找到4的根节点就能够了。

查找过程: 找到4的父节点per[4]为2,不等于它本身。继续向上查找, 2它的父节点per[2]为1。不等于它本身,继续向上查找,1的父节点per[1]为1等于它本身,说明1是根节点。

这里有一个路径压缩的优化。 当我们查找到4的根节点为1时,我们直接将per[4] = 1,即直接把4连在根节点1上,并且在查找4时还会找到2,可能还有其它的节点,将这些节点的per[]通通都设置为1。这样下次再查找4的子节点所在的树时,查找次数就缩短了1.

 

这里压缩路径有两种写法,一种是递归的,一种是非递归的。

 

1> 递归:


int find(int x){
    if(x == per[x])
        return x;
    return per[x] = find(per[x]);
}

2>非递归

int find(int x){    
    int r = x;    
    if(r != per[x])        
        r = per[x];    
    int i = x, j;    
    while(i != r){        
        j = per[i];        
        per[i] = r;        
        i = j;    
    }    
    return r;
}

请读者自己模拟一下这两种压缩路径的方式有何不同。


三、合并:

合并x 和 y所在的树。 仅仅须要把当中一个树的根节点设置为令一个树根节点的子节点就可以

 

void union(int x, int y){    
    int fx = find(x);//x的根节点为fx    
    int fy = find(y);//y的根节点为fy    
    if(fx != fy)        
        per[fx] = fy;
}

可是这里有一个问题, 是把 x的根节点设置为 y根节点的子节点,还是把y的根节点设置为x根节点的子节点。

节点1和节点2是一棵树。根节点为1, 节点3是一棵树。根节点是自身为3.

 

图1.2

 

  1.3

如图1.2所看到的:

如今我们根节点3作为根节点1的子节点,此时查找2的根节点,仅仅须要查找一次。

如图1,3所看到的 

如今我们根节点1作为根节点3的子节点。此时查找2的根节点,须要先找到1,再找到3。多了一次查找。

所以这里存在一种优化。

我们能够设置一个数组rank[ ],用它来记录每一棵树的深度,合并时假设两棵树的深度不用。那么从深度(rank)小的向深度(rank)达的连边。

(但注意,压缩路径时会使树的深度发生变化。但我们不改动rank 的值)

 

int per[maxn];//记录父节点
int rank[maxn];//记录树的深度
void init(){//初始化n个节点    
    for(int i = 1; i <= n; ++i){        
        per[i] = i;        
        rank[i] = 0;
    }
}
//找到根节点,压缩路径
int find(int x){    
    if(x == per[x])        
        return x;    
    return per[x] = find(per[x]);
}

void union (int a, int b){  
    int fa = find(a);  
    int fb = find(b);  
    if(fb != fa){  
        if(rank[fa]  < rank[fb]){  
            per[fa] = fb;  
        }  
        else{  
            per[fb] = fa;  
            if(rank[fa] == rank[fb]) rank[fa]++;  
        }  
    }    
}  

三、基础例题解析:

 

例题一:HDOJ1232--畅通project【基础并查集】

题目大意:给出n个城市, m条无向路。问最少再修几条路使全部城镇都连通。

最基础的并查集问题。递归压缩路径。没有深度优化

AC代码

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <queue>
#define maxn 1100
#define INF 0x3f3f3f3f
using namespace std;

int per[1100];
int n, m;
//初始化节点
void init(){
    for(int i = 1; i <= n; ++i)
        per[i] = i;
}
//查找根节点。递归压缩路径
int find (int x){
    if(x == per[x])
        return x;
    return per[x] = find(per[x]);
}
//合并根节点
void join(int x, int y){
    int fx = find(x);
    int fy = find(y);
    if(fx != fy)
        per[fx] = fy;
}

int main (){
    while(scanf("%d", &n),n){
        scanf("%d", &m);
        init();
        int a, b;
        while(m--){
            scanf("%d%d", &a, &b);
            join(a,b);
        }
        int ans = 0;
        //推断图中有几棵树,仅仅须要推断有几个根节点就可以
        //推断方法;父节点等于本身的节点就是根节点
        for(int i = 1; i <= n; ++i){
            if(per[i] == i)
                ans++;
        }
        //把这些根节点连通,最小须要ans - 1条边
        printf("%d\n", ans - 1);
    }
    return 0;
}

例题二:HDOJ 1272--小希的迷宫【并查集 && 判环 && 推断树的个数】

考察点:推断是否成环,推断图中树的个数

AC代码:非递归压缩路径。有深度优化

 

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#define maxn 100000 + 100

int per[maxn];
int vis[maxn];
int flag;
int ran[maxn];

//查找根节点,非递归压缩路径
int find(int x) {
    int r = x;
    while(r != per[r])
        r = per[r];
    int i,j;
    i = x;
    while(i != r){
        j = per[i];
        per[i] = r;
        i = j;
    }
    return r;
}
//合并根节点。深度优化
void jion (int a, int b){
    int fa = find(a);
    int fb = find(b);
    if(fb != fa){
        if(ran[fa]  < ran[fb]){
            per[fa] = fb;
        }
        else{
            per[fb] = fa;
            if(ran[fa] == ran[fb]) ran[fa]++;
        }
    }
    else
        flag = 0;//推断是否成环
}

int main (){
    int a, b;
    while(scanf("%d%d", &a, &b) != EOF){
        if(a == -1 && b == -1)
            break;
        if(a == 0 && b == 0){
            printf("Yes\n");
            continue;
        }
        for(int i = 1; i <= 100000; ++i){
            per[i] = i;
            vis[i] = 0;
            ran[i] = 0;
        }
        vis[a] = 1, vis[b] = 1;
        flag = 1;
        jion(a, b);
        while(scanf("%d%d", &a, &b), a || b){
            vis[a] = 1;//并非全部的房间都用到了。所以须要标记一下
            vis[b] = 1;
            jion(a, b);
        }
        int ans = 0;
        for(int i = 1; i <= 100000; ++i){
            if(per[i] == i && vis[i])
                ans++;
            if(ans > 1){
                flag = 0;
                break;
            }
        }
        if(flag) printf("Yes\n");
        else printf("No\n");

    }
    return 0;
}

四、例题推荐:

HDU 4496--D-City 【并查集 && 删边】    

解析:HDU 4496
HDU 1598--find the most comfortable road【并查集 + 枚举】   

解析:HDU 1598

HDU 2473--Junk-Mail Filter 【并查集 && 删点】  

解析:HDU 2473
HDU 3635--Dragon Balls【并查集】  

解析:HDU 3635



 本人菜鸟一个,如有不正确的地方希望各位大神纠正。有关带权并查集的问题会在日后更新

posted @ 2017-05-23 21:25  lytwajue  阅读(164)  评论(0编辑  收藏  举报