想要精通算法和SQL的成长之路 - 并查集的运用和案例(省份数量)

news2024/12/22 18:27:49

想要精通算法和SQL的成长之路 - 并查集的运用

  • 前言
  • 一. 并查集的使用和模板
    • 1.1 初始化
    • 1.2 find 查找函数
    • 1.3 union 合并集合
    • 1.4 connected 判断相连性
    • 1.5 完整代码
  • 二. 运用案例 - 省份数量

前言

想要精通算法和SQL的成长之路 - 系列导航

一. 并查集的使用和模板

先说一下并查集的相关知识点:

  • 含义:并查集,用于维护一组不相交的集合,支持合并两个集合和查询某个元素所属的集合。
  • 用途:解决图论、连通性问题和动态连通性等问题。

通俗一点,可以使用并查集的算法题目有哪些特征?

  • 需要将n个不同的元素划分为不相交的集合。
  • 开始的时候,每个元素自行成为一个集合,然后需要根据一定的顺序进行 合并
  • 同时还需要 查询 某个元素是否属于哪个集合。

因此并查集的基本操作可以包含两个:

  • 合并:将两个不相交的集合合并成一个集合。(将其中一个集合的根节点连接到另一个集合的根节点上)
  • 查找:根据某个元素,寻找到它所在集合的根节点。

1.1 初始化

首先我们考虑下,并查集里面需要有哪些数据结构:

  • 需要一个parent[]数组,用来存储每个元素对应的根节点。
  • 再来一个rank[]数组,代表以每个元素作为根节点,其所在集合的大小。即代表某个集合的深度。
  • 再来一个sum字段,代表当前的集合个数。
public class UnionFind {
    /**
     * 表示节点i的父节点
     */
    private int[] parent;
    /**
     * 表示以节点i为根节点的子树的深度,初始时每个节点的深度都为0
     */
    private int[] rank;
    private int sum;

    public UnionFind(int n) {
        parent = new int[n];
        rank = new int[n];
        // 初始时每个节点的父节点都是它自己
        for (int i = 0; i < n; i++) {
            parent[i] = i;
        }
        sum = n;
    }
}

1.2 find 查找函数

特征:

  • 入参:元素x
  • 要做的事情:不断地向上递归寻找这个x的根节点。
  • 递归终止条件:找到根节点。(根节点和元素本身一致)

代码如下:

public int find(int x) {
    while (x != parent[x]) {
        x = parent[x];
    }
    return x;
}

1.3 union 合并集合

特征:

  • 入参:元素xy
  • 要做的事情:分别找到这两个元素的根节点:rootXrootY
  • 如果俩元素的根节点是同一个,说明他们在一个集合当中,不需要任何操作。
  • 倘若两个元素的根节点不一样,根据两个集合的深度来判断。将深度小的那个集合,合并到深度大的集合中。同时更新对应的根节点和深度大小。

除此之外,我们还可以写一个简单的函数,用来判断两个元素是否处于同一个集合当中(或者是是否相连)

public void union(int x, int y) {
    int rootX = find(x);
    int rootY = find(y);
    // 如果两个元素的根节点一致,不需要合并
    if (rootX == rootY) {
        return;
    }
    // 如果根节点 rootX 的深度 > rootY。
    if (rank[rootX] > rank[rootY]) {
        // 那么将以rootY作为根节点的集合加入到rootX对应的集合当中
        rank[rootX] += rank[rootY];
        // 同时改变rootY的根节点,指向rootX。
        parent[rootY] = rootX;
    } else {
        // 反之
        rank[rootY] += rank[rootX];
        parent[rootX] = rootY;
    }
}

1.4 connected 判断相连性

/**
 * 判断两个节点是否在同一个集合中
*/
public boolean connected(int x, int y) {
    return find(x) == find(y);
}

1.5 完整代码

/**
 * @author Zong0915
 * @date 2023/10/4 下午2:52
 */
public class UnionFind {
    /**
     * 表示节点i的父节点
     */
    private int[] parent;
    /**
     * 表示以节点i为根节点的子树的深度,初始时每个节点的深度都为0
     */
    private int[] rank;
    private int sum;

    public UnionFind(int n) {
        parent = new int[n];
        rank = new int[n];
        // 初始时每个节点的父节点都是它自己
        for (int i = 0; i < n; i++) {
            parent[i] = i;
        }
        sum = n;
    }

    public int find(int x) {
        while (x != parent[x]) {
            x = parent[x];
        }
        return x;
    }

    public void union(int x, int y) {
        int rootX = find(x);
        int rootY = find(y);
        // 如果两个元素的根节点一致,不需要合并
        if (rootX == rootY) {
            return;
        }
        // 如果根节点 rootX 的深度 > rootY。
        if (rank[rootX] > rank[rootY]) {
            // 那么将以rootY作为根节点的集合加入到rootX对应的集合当中
            rank[rootX] += rank[rootY];
            // 同时改变rootY的根节点,指向rootX。
            parent[rootY] = rootX;
        } else {
            // 反之
            rank[rootY] += rank[rootX];
            parent[rootX] = rootY;
        }
    }

    /**
     * 判断两个节点是否在同一个集合中
     */
    public boolean connected(int x, int y) {
        return find(x) == find(y);
    }
}

二. 运用案例 - 省份数量

原题链接
在这里插入图片描述
我们在并查集模板的基础上进行改造:

class UnionFind {
    private int[] rank;// 每个省份具有的城市数量
    private int[] parent;// 每个城市对应的根节点(省份)
    private int sum;// 省份的数量

    public UnionFind(int[][] isConnected) {
        int len = isConnected.length;
        // 初始化,省份数量和提供的城市数量一致
        sum = len;
        // 每个集合具有的城市数量为1
        rank = new int[len];
        parent = new int[len];
        Arrays.fill(rank, 1);
        // 根节点指向自己
        for (int i = 0; i < len; i++) {
            parent[i] = i;
        }
    }

    public int find(int x) {
        while (x != parent[x]) {
            x = parent[x];
        }
        return x;
    }

    public void union(int x, int y) {
        int rootX = find(x);
        int rootY = find(y);
        // 如果两个元素的根节点一致,不需要合并
        if (rootX == rootY) {
            return;
        }
        // 如果根节点 rootX 的深度 > rootY。
        if (rank[rootX] > rank[rootY]) {
            // 那么将以rootY作为根节点的集合加入到rootX对应的集合当中
            rank[rootX] += rank[rootY];
            // 同时改变rootY的根节点,指向rootX。
            parent[rootY] = rootX;
        } else {
            // 反之
            rank[rootY] += rank[rootX];
            parent[rootX] = rootY;
        }
        // 合并成功,那么总集合数量要减1
        sum--;
    }
}

不过本题目当中,对于rank这个属性没有什么作用,最终看的是sum属性。因此大家可以把这个属性相关的给去除。

最后来看代码部分:

public int findCircleNum(int[][] isConnected) {
	// 初始化构造
    UnionFind unionFind = new UnionFind(isConnected);
    int len1 = isConnected.length;
    int len2 = isConnected[0].length;
    for (int i = 0; i < len1; i++) {
        for (int j = 0; j < len2; j++) {
            // 如果是相连的,那么将城市 i 和 j 合并
            if (isConnected[i][j] == 1) {
                unionFind.union(i, j);
            }
        }
    }
    // 最后返回集合个数(即省份的个数)
    return unionFind.sum;
}

最终代码如下:

public class Test547 {
    public int findCircleNum(int[][] isConnected) {
        UnionFind unionFind = new UnionFind(isConnected);
        int len1 = isConnected.length;
        int len2 = isConnected[0].length;
        for (int i = 0; i < len1; i++) {
            for (int j = 0; j < len2; j++) {
                // 如果是相连的,那么将城市 i 和 j 合并
                if (isConnected[i][j] == 1) {
                    unionFind.union(i, j);
                }
            }
        }
        return unionFind.sum;
    }

    class UnionFind {
        private int[] rank;// 每个省份具有的城市数量
        private int[] parent;// 每个城市对应的根节点(省份)
        private int sum;// 省份的数量

        public UnionFind(int[][] isConnected) {
            int len = isConnected.length;
            // 初始化,省份数量和提供的城市数量一致
            sum = len;
            // 每个集合具有的城市数量为1
            rank = new int[len];
            parent = new int[len];
            Arrays.fill(rank, 1);
            // 根节点指向自己
            for (int i = 0; i < len; i++) {
                parent[i] = i;
            }
        }

        public int find(int x) {
            while (x != parent[x]) {
                x = parent[x];
            }
            return x;
        }

        public void union(int x, int y) {
            int rootX = find(x);
            int rootY = find(y);
            // 如果两个元素的根节点一致,不需要合并
            if (rootX == rootY) {
                return;
            }
            // 如果根节点 rootX 的深度 > rootY。
            if (rank[rootX] > rank[rootY]) {
                // 那么将以rootY作为根节点的集合加入到rootX对应的集合当中
                rank[rootX] += rank[rootY];
                // 同时改变rootY的根节点,指向rootX。
                parent[rootY] = rootX;
            } else {
                // 反之
                rank[rootY] += rank[rootX];
                parent[rootX] = rootY;
            }
            sum--;
        }
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1059837.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

记住这份软件测试八股文还怕不能拿offer?你值得拥有

前言 2023秋招即将来临&#xff0c;很多同学会问软件测试面试八股文有必要背吗&#xff1f; 我的回答是&#xff1a;很有必要。你可以讨厌这种模式&#xff0c;但你一定要去背&#xff0c;因为不背你就进不了大厂。 国内的互联网面试&#xff0c;恐怕是现存的、最接近科举考试…

计算机竞赛 车道线检测(自动驾驶 机器视觉)

0 前言 无人驾驶技术是机器学习为主的一门前沿领域&#xff0c;在无人驾驶领域中机器学习的各种算法随处可见&#xff0c;今天学长给大家介绍无人驾驶技术中的车道线检测。 1 车道线检测 在无人驾驶领域每一个任务都是相当复杂&#xff0c;看上去无从下手。那么面对这样极其…

【MySQL】表的约束(一)

文章目录 为什么要有约束一. 空属性二. 默认值三. 列描述四. zerofill结束语 为什么要有约束 数据库是用来存放数据的&#xff0c;所以其需要保证数据的完整性和可靠性 数据类型也算是一种约束&#xff0c;比如&#xff0c;整型的数据无法插入字符型。 通过约束&#xff0c;让…

超详细!手把手带你实现一个完整的Promise

Promise是JavaScript中异步编程的解决方案&#xff0c;一开始在社区中提出和实现&#xff0c;后来ECMAScript将其写进了标准中。Promise有效的解决了异步编程的回调地狱问题&#xff0c;非常受开发者的欢迎。 本文首先介绍了JavaScript中异步编程的几种方式&#xff0c;再对Pr…

Vue学习之页面上中下三层布局

Vue学习之页面上中下三层布局 页面布局&#xff1a;头部&#xff0c;内容区&#xff0c;尾部&#xff0c;其中头部和尾部几乎所有页面都有&#xff0c;可抽成公共组件&#xff0c;内容区是可变的&#xff0c;由路由组件展示 页面效果 实现 &#xff08;1&#xff09;app.vue &…

为什么在使用PageHelper插件时,指定的每页记录数大小失效?显示所有的记录数

1.问题现象&#xff1a; 这里指定每页显示5条&#xff0c;却把所有的记录数都显示出来了 2.分析&#xff1a; 之前是可以的&#xff0c;然后发现&#xff1a;PageHelper.startPage(pageNum,pageSize) 和执行sql的语句 顺序颠倒了&#xff0c;然后就出错了。 3.验证&#xf…

十天学完基础数据结构-第四天(链表(Linked List))

链表的基本概念 链表是一种线性数据结构&#xff0c;与数组不同&#xff0c;链表的元素&#xff08;节点&#xff09;之间通过指针相互连接。链表有以下基本概念&#xff1a; 节点&#xff1a;链表中的每个数据项称为节点&#xff0c;每个节点包含数据和一个指向下一个节点的指…

【Unity2022】Unity实现手机游戏操控摇杆(实现操控轮盘)

文章目录 运行效果预览创建物体脚本获取RectTransform处理玩家拖动事件完整代码 获取输入运行其他文章 运行效果预览 首先展示一下本文章实现的效果&#xff1a; 创建物体 创建两个UI图像&#xff0c;一个用于表示背景&#xff0c;作为父物体&#xff0c;命名为JoyStick&am…

ubuntu安装ROS rosdep init rosdep update报错,完美解决方案!

ubuntu安装ROS rosdep init rosdep update报错&#xff0c;终于让我发现完美解决方法啦&#xff01;清华源解决 问题的原因完美解决&#xff01; 问题的原因 rosdep init&#xff0c;rosdep update报错的根本原因还是国内网络连不上外网。所以改DNS之类的方法都是比较偶然能成…

Windows安装Docker并创建Ubuntu环境及运行神经网络模型

目录 前言在Windows上安装Docker在Docker上创建Ubuntu镜像并运行容器创建Ubuntu镜像配置容器&#xff0c;使其可以在宿主机上显示GUI 创建容器并运行神经网络模型创建容器随便找一个神经网络模型试试 总结 前言 学生党一般用个人电脑玩神经网络&#xff0c;估计很少有自己的服…

nginx下载与安装教程

文章目录 nginx简介nginx的主要应用场景nginx开源项目的源码结构 使用centos7安装nginx检查centos版本号和linux内核版本检查是否安装gcc、pcre、zlib、openssl等依赖 安装nginx启动nginx停止nginx重启nginx nginx简介 nginx是一款业内流行、功能强大的web服务器。 高性能&…

会声会影2024中文版好用吗?

近些年&#xff0c;短视频逐渐走红并普及到各个领域&#xff0c;吸引着大量的自媒体从业者和爱好者投身于视频制作的热潮之中。视频剪辑软件作为视频制作不可或缺的工具&#xff0c;那么如何选择视频剪辑软件呢&#xff1f;视频剪辑软件哪个好&#xff1f; 一、视频剪辑软件有哪…

2023年,在CSDN拥有10000粉丝有多难?

该数据来源于粉丝数人数排行前5000名用户的关注用户列表中产生的&#xff0c;由于采集样本数有限&#xff0c;数据可能具有一定的误差&#xff0c;仅供参考&#xff0c;本次采样用户数大概在100万以上。 筛选条件人数粉丝人数大于50007519粉丝人数大于100003763粉丝人数大于500…

GJB 128B-2021标准版本变更汇总 ,发布, 下载

GJB 128B-2021标准版&#xff0c;下载 一、 概述 GJB 128B-2021半导体分立器件试验方法已于2022年3月1日实施&#xff0c;相对现行标准&#xff0c;新版标准对部分内容进行了变更。针对我司VDMOS产品涉及的各种方法&#xff0c;我司对新标准与旧标准的差异进行了分析。 二、 …

【AI视野·今日NLP 自然语言处理论文速览 第四十七期】Wed, 4 Oct 2023

AI视野今日CS.NLP 自然语言处理论文速览 Wed, 4 Oct 2023 Totally 73 papers &#x1f449;上期速览✈更多精彩请移步主页 Daily Computation and Language Papers Contrastive Post-training Large Language Models on Data Curriculum Authors Canwen Xu, Corby Rosset, Luc…

国庆中秋宅家自省: Python在Excel中绘图尝鲜

【一】国庆中秋: 悟 【国庆中秋】双节来临,相信各位有自己度过的方式,而我却以独特的方式度过了一个说出来不怕各位见笑的双节; 双节到来,没有太多惊喜&#xff0c;也没有太多的负面情绪, 只是喜欢独处,静静反省这些年走过的酸甜苦辣&#xff1b;生活中的许多不欢而散,不期而遇…

反素数

198. 反素数 - AcWing题库 最大的反素数也就是约数个数最多的数中最小的那个数&#xff0c;可以考虑分解质因子形式 2、3、5、7、11、13、17、19、23、29这些&#xff0c;还有每个质因子的指数一定大于等于下一个质因子的指数&#xff0c;这样可以保证约数最多的时候数字尽可能…

基于SpringBoot的智能推荐的卫生健康系统

目录 前言 一、技术栈 二、系统功能介绍 用户管理 科室类型管理 医生信息管理 健康论坛管理 我的发布 我的收藏 在线咨询 三、核心代码 1、登录模块 2、文件上传模块 3、代码封装 前言 随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理信息系统的实施在…

【神印王座】陈樱儿假扮魔神皇,皓晨想杀人灭口,采儿施展禁制,月夜成功自保

Hello,小伙伴们&#xff0c;我是小郑继续为大家深度解析国漫资讯。 神印王座动画即将更新&#xff0c;官方早早就公布了最新集的预告。虽然三大荒野部族已经全都被灭了&#xff0c;但是危险并没有解除&#xff0c;陈樱儿假扮魔神皇救人。逃出生天后&#xff0c;猎魔团与月夜商会…

数字电路逻辑 之 逻辑与逻辑运算

注意逻辑顺序&#xff0c; * 是大于 左边这一列与右边这一列是对偶关系&#xff0c;真值相同