数据结构之并查集

简介: 并查集

并查集


并查集被认为是最简洁而优雅的数据结构之一,主要用于解决一些元素分组的问题。它管理一系列不相交的集合,并支持两种操作:

  • 合并(Union):把两个不相交的集合合并为一个集合。
  • 查询(Find):查询两个元素是否在同一个集合中。

1654597029927.png


Quick Find


第一个版本的并查集


public interface UnionFind {
        boolean isConnected(int p,int q);
        void unionElements(int p ,int q);
        int getSize();
}
public class UnionFind1 implements UnionFind {
        private int[] id;
        public UnionFind1(int size) {
            id = new int[size];
            for (int i = 0; i < id.length; i++) {
                id[i] = i;
            }
        }
        //查看元素p和元素q是否属于一个集合
        @Override
        public boolean isConnected(int p, int q) {
            return find(p) == find(q);
        }
        //合并元素p和元素q所属的集合
        @Override
        public void unionElements(int p, int q) {
            int pID = find(p);
            int qID = find(q);
            if (pID == qID) {
                return;
            }
            for (int i = 0; i < id.length; i++) {
                if (id[i] == pID) {
                    id[i] = qID;
                }
            }
        }
        @Override
        public int getSize() {
            return id.length;
        }
        // 查找元素p所对应的集合编号
        private int find(int p) {
            if (p < 0 || p >= id.length) {
                throw new IllegalArgumentException("p is out of bound.");
            }
            return id[p];
        }
}
操作 时间复杂度
unionElements(p,q) O(n)
isConnected(p,q) O(1)


Quick Union


我们可以将每一个元素都看作是一个节点。


1654596810788.png


如果节点3想要连接节点2,那就是节点3去连接节点2,而2又指向自己


1654596968284.png


如果节点1想要连接节点3也是需要连接节点2即可。如果另一个节点的7想要连接2也是需要当前节点的根节点去连接2即可。


1654596860300.png


一开始的时候使用数组表示,每一个节点都是根节点和其他节点无关联。


1654596914363.png


如果我们想union 4,3 节点,我们只需要让4指向3即可。


1654596968284.png

1654597029927.png


如果想union3,8其实也非常简单,只需要用3指向8的节点


1654597184700.png



如果想union9,4其实并不是指向4这个节点,而是指向4的根节点的8。


1654597227787.png

public class UnionFind2 implements UnionFind{
    private int[] parent;
    public UnionFind2(int size) {
        parent = new int[size];
        for (int i = 0; i < size; i++) {
            parent[i] = i;
        }
    }
    //查看元素p和元素q是否属于一个集合
    // O(h)复杂度,h为树的高度
    @Override
    public boolean isConnected(int p, int q) {
        return find(p) == find(q);
    }
    //合并元素p和元素q所属的集合
    // O(h)复杂度,h为树的高度
    @Override
    public void unionElements(int p, int q) {
        int pRoot = find(p);
        int qRoot = find(q);
        if (pRoot == qRoot) {
            return;
        }
        parent[pRoot] = qRoot;
    }
    @Override
    public int getSize() {
        return parent.length;
    }
    //查找过程,查找元素p所对应的集合编号
    // O(h)复杂度,h为树的高度
    private int find(int p){
        if (p < 0 || p >= parent.length) {
            throw new IllegalArgumentException("p is out of bound.");
        }
        while(p != parent[p]){
            p = parent[p];
        }
        return p;
    }
}
操作 时间复杂度
unionElements(p,q) O(h)
isConnected(p,q) O(h)


基于Size的优化


如果我们union 0,1 然后 union 0,2 然后 union 0,3这样的话就会产生一定的问题,因为我们没有对合并的元素的树没有做判断,所以会导致我们不断增加树的高度,从而成链表的结构。


1654597467970.png


如果我们的树是这样子的。


1654597503150.png


我们想union 4,9的话,我们的树就会变成这个样子。深度就达到了4。


1654597533789.png


但其实我们可以让9来指向4的根节点也就是8。这样我们的深度就只有3。


1654597636833.png

public class UnionFind3 implements UnionFind{
    private int[] parent;
    //sz[i] 表示以i为根的集合中元素个数
    private int[] sz;
    public UnionFind3(int size) {
        parent = new int[size];
        sz = new int[size];
        for (int i = 0; i < size; i++) {
            parent[i] = i;
            sz[i] = 1;
        }
    }
    //查看元素p和元素q是否属于一个集合
    // O(h)复杂度,h为树的高度
    @Override
    public boolean isConnected(int p, int q) {
        return find(p) == find(q);
    }
    //合并元素p和元素q所属的集合
    // O(h)复杂度,h为树的高度
    @Override
    public void unionElements(int p, int q) {
        int pRoot = find(p);
        int qRoot = find(q);
        if (pRoot == qRoot) {
            return;
        }
        if(sz[pRoot] < sz[qRoot]){
            parent[pRoot] = qRoot;
            sz[qRoot] += sz[pRoot];
        }else{
            parent[qRoot] = pRoot;
            sz[pRoot] += sz[qRoot];
        }
    }
    @Override
    public int getSize() {
        return parent.length;
    }
    //查找过程,查找元素p所对应的集合编号
    // O(h)复杂度,h为树的高度
    private int find(int p){
        if (p < 0 || p >= parent.length) {
            throw new IllegalArgumentException("p is out of bound.");
        }
        while(p != parent[p]){
            p = parent[p];
        }
        return p;
    }
}


基于Rank的优化


1654597674899.png


假设现在有这样一棵树,我们进行union4,2根据size优化我们会把8来指向7。现在深度就变为了4。


1654597703476.png


但是这样子的话,原本的高度是2一下就变为了4,为了优化其实我们可以将7来指向8。深度就变为了3。我们需要将深度低的指向深度高的树


1654597859947.png

public class UnionFind4 implements UnionFind{
    private int[] parent;
    //rank[i] 表示以i为根的集合中树的层数
    private int[] rank;
    public UnionFind4(int size) {
        parent = new int[size];
        rank = new int[size];
        for (int i = 0; i < size; i++) {
            parent[i] = i;
            rank[i] = 1;
        }
    }
    //查看元素p和元素q是否属于一个集合
    // O(h)复杂度,h为树的高度
    @Override
    public boolean isConnected(int p, int q) {
        return find(p) == find(q);
    }
    //合并元素p和元素q所属的集合
    // O(h)复杂度,h为树的高度
    @Override
    public void unionElements(int p, int q) {
        int pRoot = find(p);
        int qRoot = find(q);
        if (pRoot == qRoot) {
            return;
        }
        //根据两个元素所在树的rank不同判断合并方向
        //将rank低的集合合并到rank高的集合上
        if(rank[pRoot] < rank[qRoot]) {
            parent[pRoot] = qRoot;
        }else if(rank[qRoot] < rank[pRoot]){
            parent[qRoot] = pRoot;
        }else{
            parent[qRoot] = pRoot;
            rank[pRoot] += 1;
        }
    }
    @Override
    public int getSize() {
        return parent.length;
    }
    //查找过程,查找元素p所对应的集合编号
    // O(h)复杂度,h为树的高度
    private int find(int p){
        if (p < 0 || p >= parent.length) {
            throw new IllegalArgumentException("p is out of bound.");
        }
        while(p != parent[p]){
            p = parent[p];
        }
        return p;
    }
}

路径压缩


下图中三种树的操作其实都是一样的,但是左边的深度达到了5,而中间可只有2。我们应该如何进行路径压缩?


1654597886347.png


这样的一棵树结构下,如果我们进行下面操作

parent[p] = parent[parent[p]];

1654597907750.png


我们让4节点来指向父节点的父节点,也就变成了下面这样。


1654597947050.png


然后我们再让4的父节点执行同样操作就会变成这样。


1654597973791.png

//查找过程,查找元素p所对应的集合编号
    // O(h)复杂度,h为树的高度
    private int find(int p){
        if (p < 0 || p >= parent.length) {
            throw new IllegalArgumentException("p is out of bound.");
        }
        while(p != parent[p]){
            //路径压缩
            parent[p] = parent[parent[p]];
            p = parent[p];
        }
        return p;
    }
相关文章
|
8月前
|
容器
数据结构:并查集
数据结构:并查集
77 0
数据结构:并查集
|
3月前
|
算法 开发者 计算机视觉
燃爆全场!Python并查集:数据结构界的网红,让你的代码炫酷无比!
在编程的世界里,总有一些数据结构以其独特的魅力和高效的性能脱颖而出,成为众多开发者追捧的“网红”。今天,我们要介绍的这位明星,就是Python中的并查集(Union-Find)——它不仅在解决特定问题上大放异彩,更以其优雅的设计和强大的功能,让你的代码炫酷无比,燃爆全场!
46 0
|
8月前
|
机器学习/深度学习 存储
数据结构(九)---并查集
数据结构(九)---并查集
112 5
|
4月前
|
Python
逆天改命!掌握Python并查集,数据结构难题从此不再是你的痛!
在编程旅程中,遇到棘手的数据结构难题是否让你苦恼?别担心,Python并查集(Union-Find)是你的得力助手。这是一种高效处理不相交集合合并及查询的数据结构,广泛应用于网络连通性、社交网络圈子划分等场景。通过维护每个集合的根节点,它实现了快速合并与查询。本文将介绍并查集的基本概念、应用场景以及如何在Python中轻松实现并查集,帮助你轻松应对各种数据结构挑战。
43 3
|
4月前
|
存储 C语言
数据结构基础详解(C语言): 树与二叉树的应用_哈夫曼树与哈夫曼曼编码_并查集_二叉排序树_平衡二叉树
本文详细介绍了树与二叉树的应用,涵盖哈夫曼树与哈夫曼编码、并查集以及二叉排序树等内容。首先讲解了哈夫曼树的构造方法及其在数据压缩中的应用;接着介绍了并查集的基本概念、存储结构及优化方法;随后探讨了二叉排序树的定义、查找、插入和删除操作;最后阐述了平衡二叉树的概念及其在保证树平衡状态下的插入和删除操作。通过本文,读者可以全面了解树与二叉树在实际问题中的应用技巧和优化策略。
|
4月前
|
Python
告别低效!Python并查集:数据结构界的超级英雄,拯救你的编程人生!
告别低效!Python并查集:数据结构界的超级英雄,拯救你的编程人生!
35 0
|
4月前
|
算法 开发者 计算机视觉
Python并查集:数据结构界的肌肉男,让你在编程路上无所畏惧!
在编程的浩瀚宇宙中,数据结构如同基石,构建了解决问题的坚实框架。而并查集(Union-Find),这位数据结构界的“肌肉男”,以其独特的魅力和强大的功能,让无数开发者在面对复杂关系处理时,都能感受到前所未有的从容与自信。今天,就让我们一同揭开并查集的神秘面纱,看看它是如何成为你编程路上的得力助手的。
38 0
|
4月前
|
算法 程序员 计算机视觉
Python并查集:数据结构界的肌肉男,让你在编程路上无所畏惧!
并查集,一种处理不相交集合合并与查询的数据结构,被誉为编程的“肌肉男”。它提供Find(找根节点)和Union(合并集合)操作,常用于好友关系判断、图像处理、集合合并等。Python实现中,路径压缩和按秩合并优化效率。并查集的高效性能使其成为解决问题的强大工具,助力程序员应对复杂挑战。
39 0
|
6月前
|
算法 程序员 图形学
脑洞大开!Python并查集:用最简单的方式,解决最复杂的数据结构问题!
【7月更文挑战第17天】并查集,数据结构明星,处理不相交集合合并与查询。Python实现核心操作:查找与合并。路径压缩优化查找,按秩合并保持平衡。实战应用如图连通性判断,算法竞赛利器。掌握并查集,解锁复杂问题简单解法,照亮编程之旅!
66 10
|
6月前
|
Python
告别低效!Python并查集:数据结构界的超级英雄,拯救你的编程人生!
【7月更文挑战第18天】并查集,数据结构超级英雄,用于不相交集合的合并与查询。Python实现包括初始化、查找根节点和合并操作。应用广泛,如社交网络分析、图论问题、集合划分等。示例代码展示了解决岛屿数量问题,统计连通的“1”单元格数。掌握并查集,提升编程效率,解决复杂问题。
59 6