《六月集训》(第二十六天)——并查集

前言

        欢迎大家积极在评论区留言发表自己的看法,知无不言,言无不尽,养成每天刷题的习惯,也可以自己发布优质的解题报告,供社区一同鉴赏,吸引一波自己的核心粉丝。
        今天是六月集训第二十六天:并查集
在这里插入图片描述

一、练习题目

        1559. 二维网格图中探测环
        1391. 检查网格中是否存在有效路径
        1202. 交换字符串中的元素

二、算法思路

1.并查集原理

还在学并查集。

1.1并查集是一种树型的数据结构,用于处理一些不相交集合(disjoint sets)的合并及查询问题。
1.2并查集通常包含两种操作

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

2.适用说明

并查集用在一些N个元素的集合应用问题中,我们通常是在开始时让每个元素构成一个单元素的集合,然后按一定顺序将属于同一组的元素所在的集合合并,其间要反复查找一个元素在哪个集合中。这个过程看似并不复杂,但数据量极大,若用其他的数据结构来描述的话,往往在空间上过大,计算机无法承受,也无法在短时间内计算出结果,所以只能用并查集来处理。

3.并查集的基本数据表示

如上图 0-4 下面都是 0,5-9 下面都是 1,表示 0、1、2、3、4 这五个元素是相连接的,5、6、7、8、9 这五个元素是相连的。

4.并查集C++初始化实现

const int MAXN = 300010;
int fset[MAXN];

void init(int n) {
    // 初始化每一个,fset[i]指向自己,没有涉及到合并的操作。
    for (int i = 0; i < n; ++i) {
        fset[i] = i;
    }
}

现在每个结点各自为王,所以自己就是自己的老大,所以Parent数组指向的就是自己本身。

5.并查集查找C++实现

O(1)的时间复杂度,查询元素所在的集合编号

// 查询根结点
int find(int x){
    if(Parent[x]==x)
        return x;
    else
        return find(Parent[x]);
}

查找自己的老大是谁,用递归的方法实现对集合中的代表元素(老大)的查询:一层一层访问双亲结点,直至根结点(根结点的标志就是根结点本身)。要判断两个元素是否属于同一个集合,只需要看它们的根结点是否相同即可,即看他们的老大是否是同一个人。

例如查找鬼灯水月的老大是谁?

通过递归查找,先查找数组下标为2的元素,为1,所以他自己不是自己的老大,他的老大是1号,也就是佐助
接下来看看佐助的老大是谁,查看数组下标为1的元素,为0,所以佐助也不是自己的老大,佐助的老大是0号,也就是鸣人
接下来看看鸣人的老大是谁,查看数组下标为0的元素,为0,所以鸣人自己就是老大,所以递归结束,这个集合的老大为0号。

6.并查集的合并C++实现

void merge(int x, int y) {
    int rx = find(x), ry = find(y);
    if (rx != ry) {
        for (int i = 1; i <= n; ++i) {
            if (fset[i] == rx) {
                fset[i] = ry;
            }
        }
    }
}
  • 1、1559. 二维网格图中探测环:
  • 2、1391. 检查网格中是否存在有效路径:
  • 3、1202. 交换字符串中的元素:

三、源码剖析

// 1559. 二维网格图中探测环

  • 1、
// 1391. 检查网格中是否存在有效路径

  • 1、
// 1202. 交换字符串中的元素

  • 1、
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值