并并并并·病查坤

P1、什么是并查集

引用自百度百科:

并查集,在一些有N个元素的集合应用问题中,我们通常是在开始时让每个元素构成一个单元素的集合,然后按一定顺序将属于同一组的元素所在的集合合并,其间要反复查找一个元素在哪个集合中。

并查集是一种树型的数据结构,用于处理一些不相交集合(disjoint sets)的合并及查询问题。常常在使用中以森林来表示。

简单来说,并查集是一种以树形结构来表示不同种类数据的集合。一般当我们需要用到数据的连通性时会用到它。

并查集维护一个数组parent,parent数组中维护的不是元素本身,而是元素的下标索引,当然,这个下标索引是指向该元素的父元素的。

image-20240425223445850

引用自:【算法与数据结构】—— 并查集-CSDN博客

P2、简单、无优化的并查集

// 未改进版本
public class Djset {
    private int[] parent;  // 记录节点的根
    public Djset(int n) {
        for (int i = 0; i < n; i++) 
            parent[i] = i;
    }

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

    public void merge(int x, int y) {
        int rootx = find(x);
        int rooty = find(y);
        if (rootX != rootY) {
            parent[rootY] = rootX;
        }
    }
}

这种没有任何优化的并查集,比较简单,但是效率很低。为什么?

问题1:

当合并两个节点时,这里是没有任何判断的,便直接将rootx设置成了rooty的父节点。

假如rooty的叶子节点深度比rootx的叶子节点深度大呢?此时,树的深度会持续增加,会造成后续的节点查询时间长。

问题2:

在寻找某个节点的根节点的过程中,我们也未对其父节点和祖父节点…等作任何操作。

假如该节点会被合并(merge)很多次,而每次都要经过父节点、祖父节点…层层寻找,造成了不必要的时间浪费。

比如:

image-20240425221005247

这样,树的深度便在无形之中增加了1。

如果反过来,将rootx的父节点设置为rooty,看下效果:

image-20240425220909242

树的深度是没有增加的,不会对后续节点造成影响。

P3、优化后的并查集【按秩合并】【路径压缩】

// 注意:使用该代码,并不能使得所有的元素都直接指向根节点,仍然存在间接的指向
public class Djset {
    private int[] parent;  // 记录节点的根
    private int[] rank;  // 记录根节点的深度,优化合并操作

    // 构造函数,初始化每个节点的根为其自身,并设置初始秩为0
	public Djset(int n) {
    	parent = new int[n];
   		rank = new int[n];
  	  	for (int i = 0; i < n; i++) {
  	      	parent[i] = i;
        	rank[i] = 1; // 确保初始化每个节点的初始秩也为1
   		}
	}

    // 查找x的根节点,同时进行路径压缩
    public int find(int x) {
        if (x != parent[x]) {
            parent[x] = find(parent[x]);  // 路径压缩至根节点
        }
        return parent[x];
    }

    // 合并x和y所在的集合,按秩合并优化
    public void merge(int x, int y) {
        int rootX = find(x);
        int rootY = find(y);
        if (rootX != rootY) {
            // 按秩合并:将秩较小的集合合并到较大的集合
            if (rank[rootX] < rank[rootY]) {
                int temp = rootX;
                rootX = rootY;
                rootY = temp;
            }
            parent[rootY] = rootX;  // 根据秩决定合并方向
            if (rank[rootX] == rank[rootY]) {
                rank[rootX]++;
            }
        }
    }
}

这里主要针对未改进的版本,做了两点优化 【按秩合并】和【路径压缩】

① 按秩合并

好的,让我们先弄清楚什么是按秩合并

秩:我们暂时将其定义为节点的最大深度(从节点自身开始,到叶子节点的最大深度)

按秩合并:主要是针对merge函数,在合并两个集合时,将秩大的根节点设置为秩小的根节点的父节点。意思是当要合并两个根节点A、B时,如果节点A的秩大于节点B的秩,那么将节点A设置为节点B的父节点,反之亦然。

比如:

image-20240425220909242

通过将秩大的根节点设置为合并后的根节点,避免了树的深度增加。

② 路径压缩

同样,先弄清楚什么是路径压缩

路径压缩:主要针对find函数,当在寻找一个节点A的根节点root时,直接将节点A的父节点B、祖父节点C…等节点全部指向根节点root。

优点:这样在下次寻找A的根节点、B的根节点、C的根节点时可以节省很长一段搜索路径。

如:

image-20240425221904252

接下来,来几个简单的困难题:

原题链接:

839. 相似字符串组 - 力扣(LeetCode)

image-20240426215746045

这个题目首先要理解相似的定义:

两个字符串相似的含义是能够通过交换两个字符的位置,得到另外一个字符串。

根据题目来看,给的字符串中的字母是相同的,不同的顺序,那么相似只有两种情况:两个字符串的对应位置中只有 0 个或者 2 个不同。

我们将字符串看作节点,两重 for 循环,实现对节点之间两两组合,判断两个节点是否相似,判断相似的方法是:

两个字符串的对应位置中只有 0 个或者 2 个不同;

如果两个字符串相似则将他们归入一组,之后遍历时,如果不是同一组就需要进行判断。

初始化并查集数组,并初始化集合数,每次合并减一。

class Solution {
    public static int[] father = new int[10000];
    public static int sets;
    public static void build(int n) {
        for (int i = 0; i < n; i++) {
            father[i] = i;
        }
        sets = n;
    }
    public static int find(int i) {
        if (i != father[i]) {
            father[i] = find(father[i]);
        }
        return father[i];
    }
    public static void union(int x, int y) {
        int fx = find(x);
        int fy = find(y);
        if (fx != fy) {
            father[fx] = fy;
            sets--;
        }
    }
    public int numSimilarGroups(String[] strs) {
        int n = strs.length;
        int m = strs[0].length();
        build(n);
        for (int i = 0; i < n; i++) {
            for (int j = i + 1; j < n; j++) {
                if (find(i) != find(j)) {
                    int diff = 0;
                    for (int k = 0; k < m && diff < 3; k++) {//不同之处大于三那就没有意义了,一定不相似
                      
                        if (strs[i].charAt(k) != strs[j].charAt(k)) {
                            diff++;
                        }
                    }
                    if (diff == 0 || diff == 2) {
                        union(i, j);
                    }
                }
            }
        }
        return sets;
    }
}

原题链接:

765. 情侣牵手 - 力扣(LeetCode)

image-20240425204217850

首先,我们总是以「情侣对」为单位进行设想:

当有两对情侣相互坐错了位置,ta们两对之间形成了一个环。需要进行一次交换,使得每对情侣独立(相互牵手)

长度1的自环

如果三对情侣相互坐错了位置,ta们三对之间形成了一个环,需要进行两次交换,使得每对情侣独立(相互牵手)

image.png

如果四对情侣相互坐错了位置,ta们四对之间形成了一个环,需要进行三次交换,使得每对情侣独立(相互牵手)

也就是说,如果我们有 k 对情侣形成了错误环,需要交换 k - 1 次才能让情侣牵手。

于是问题转化成 n / 2 对情侣中,有多少个这样的环。

如果他们本来就是情侣,他们处于大小为1的错误环中,只需要交换0次。

如果不是情侣,说明他们呢两对处在同一个错误环中,我们将他们合并,将所有的错坐情侣合并后,答案就是情侣对 - 环个数。

class Solution {
    public int minSwapsCouples(int[] row) {
        int n = row.length;
        bulid(n / 2);
        for (int i = 0; i < n; i += 2) {
            union(row[i] / 2, row[i + 1] / 2);

        }
        return n / 2 - set;
    }
    //首先计算数组长度的一半(即情侣对数),然后调用bulid方法初始化并查集,
    //之后遍历数组,对每一对执行union操作合并它们所在的集合。
   // 最后,返回初始集合数量减去最终集合数量的结果,即为最少需要交换的次数。

    public static int[] father = new int[100];
    public static int set;

    public static void bulid(int x) {
        set = x;
        for (int i = 0; i < x; i++) {
            father[i] = i;
        }
    }
    public static int find(int i){
        if(i==father[i])return father[i];
        else return find(father[i]);
    }

    public static void union(int x,int y){
        if(find(x)!=find(y)){
            father[find(x)]=find(y);
            set--;
        }
    }
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

虹科Pico汽车示波器 | 免拆诊断案例 | 2006 款林肯领航员车发动机怠速抖动

故障现象 一辆2006款林肯领航员车&#xff0c;搭载5.4 L发动机&#xff0c;累计行驶里程约为26万km。该车因发动机怠速抖动故障进厂维修&#xff0c;维修人员更换了火花塞、点火线圈及凸轮轴位置传感器&#xff0c;清洗了积炭和喷油器&#xff0c;故障依旧&#xff0c;于是向笔…

纵览2024年:排名靠前的项目管理软件一览!

时间飞逝&#xff0c;2024年已经过去近半&#xff0c;让我们来盘点2024年排名靠前的项目管理软件&#xff0c;项目管理软件排行榜&#xff0c;本次上榜的项目管理软件有Zoho Projects、Microsoft Project、Nifty、Smartsheet、ClickUp。 一、项目管理软件排行榜 1.Zoho Projec…

8点法估计基础矩阵

估计基础矩阵 文章目录 估计基础矩阵8点法归一化 8点法 8点法 根据两幅图像中8个对应点对之间的关系&#xff0c;采用SVD求 解最小二乘方 约束&#xff1a;det(F) 0 假设已知N对点的对应关系&#xff1a; { x i , x i ′ } i 1 N \{x_i,x^{\prime}_i\}_{i1}^N {xi​,xi′​…

RKNN:yolov8模型转换与板端推理流程

近期&#xff0c;在研究瑞芯微的RKNN模型推理时&#xff0c;遇到一些坑&#xff0c;现记录下来&#xff0c;以备忘&#xff0c;亦供同道者参考。 目录 1. 模型转换 1.1. 宿主机环境配置 1.2. onnx模型准备 1.3. onnx转rknn 2. 模型推理 2.1. 推理环境配置 2.2. 推理验证…

码农解压宝典

在快速发展的IT行业中&#xff0c;程序员们面临着巨大的工作压力。长时间的工作、高强度的编程任务以及不断更新的技术知识&#xff0c;使得程序员们时常感到疲惫不堪。然而&#xff0c;通过掌握一些简单的小窍门&#xff0c;程序员们可以有效地缓解工作压力&#xff0c;保持身…

【C++】类和对象⑤(static成员 | 友元 | 内部类 | 匿名对象)

&#x1f525;个人主页&#xff1a;Forcible Bug Maker &#x1f525;专栏&#xff1a;C 目录 前言 static静态成员 友元 友元函数 友元类 内部类 匿名对象 结语 前言 本篇主要内容&#xff1a;类和对象的一些知识点补充&#xff0c;包括static静态成员&#xff0c;友…

AWTK 开源串口屏开发(17) - 通过 MODBUS 访问数组数据

在 AWTK 串口屏中&#xff0c;内置了 MODBUS Client Channel 的模型&#xff0c;不用编写代码即可实现在 ListView 中显示数组数据。 MODBUS 协议一次只能读取 125 个 WORD&#xff0c;AWTK-MODBUS Client Channel 支持长数据&#xff0c;自动分成多个请求访问。 1. 功能 不用…

C语言入门课程学习记录5

C语言入门课程学习记录5 第23课 - C 语言中的常量第24课 - 初探程序中的数组第25课 - 数组特性深入剖析第26课 - 多维数组的概念与示例 本文学习自狄泰软件学院 唐佐林老师的 C语言入门课程&#xff0c;图片全部来源于课程PPT&#xff0c;仅用于个人学习记录 第23课 - C 语言中…

SecretFlow学习指南(2)学习路径

目录 一、模块架构 二、模块详解 三、算法协议 四、学习路线 一、模块架构 良好的分层设计可以提高开发效率和可维护性&#xff0c;满足不同用户的需求。隐语从上到下一共分为六层。 ●产品层&#xff1a;通过白屏化产品提供隐语整体隐私计算能力的输出&#xff0c;让用户简…

paddle ocr模型量化实践

参考&#xff1a;https://github.com/PaddlePaddle/PaddleOCR/blob/main/deploy/slim/quantization/README.md https://github.com/PaddlePaddle/PaddleOCR/blob/release/2.7.1/doc/doc_ch/FAQ.md 蒸馏 剪枝 量化 参考&#xff1a;https://blog.csdn.net/mddCSDN/article/de…

【MySQL】MySQL中MVCC多版本并发控制的概念

MySQL中MVCC多版本并发控制的概念 锁相关的知识我们已经学习完了&#xff0c;在其中我们提到过一个概念&#xff0c;那就是 MVCC 。这又是个什么东西呢&#xff1f;今天我们就来好好看看 MVCC 到底是干嘛的。 MVCC 多版本并发控制&#xff0c;它主要是控制 读 操作&#xff0c;…

x264 编码器源码分析综述

================================================================================ 系列文章 x264配置文章链接🔗Windows11编译x264源码https://blog.csdn.net/yanceyxin/article/details/135035650Mac调试x264源码https://blog.csdn.net/yanceyxin/article/details

《软件设计师教程:计算机网络浅了解计算机之间相互运运作的模式》

​ 个人主页&#xff1a;李仙桎 &#x1f525; 个人专栏: 《软件设计师》 ⛺️生活的理想&#xff0c;就是为了理想的生活! ​ ⛺️前言&#xff1a;各位铁汁们好啊&#xff01;&#xff01;&#xff01;&#xff0c;今天开始继续学习中级软件设计师考试相关的内容&#xff0…

python中怎么注释多行

多行代码注释 方法一&#xff1a;先选中要注释的段落&#xff0c;然后按下“ctrl/”&#xff0c;即可实现多行代码的注释。效果如下&#xff1a; 再一次按下“ctrl/”就可以取消注释。 方法二&#xff1a;跟注释单行一样在每一行前面输入“shift#”。 #r(i-arr[idx])*rat[idx]…

三阶魔方公式大全 图解

https://www.mitao521.com/miji/2020112215034.html 三阶魔方七步还原法的公式有R’UF’U’、R’D’RD X,3OR5,R U R’,(RU R’U’),(RU R’U’)3,U’ L’ U L U F U’ F’,U R U’ R’ U’ F’ U F,F(R U R’ U’)F’。 还有(R U R’ U’)2和(R U R’ U’)5,R2 D2 R’ U’ R …

各种螺纹介绍

按用途&#xff0c;有三个主要大类&#xff1a; 第一&#xff0c;连接螺纹&#xff0c;用于紧固&#xff0c;即是螺栓螺母&#xff1b; 第二&#xff0c;传动螺纹&#xff0c;就是车床走刀那种&#xff1b; 第三&#xff0c;管螺纹&#xff0c;管道连接用。 按标准&#xf…

【刷题篇】动态规划-01背包问题(十)

文章目录 1、01背包2、分割等和子集3、目标和4、最后一块石头的重量 II 1、01背包 #include <iostream> #include<vector> using namespace std;int main() {int n,v;cin>>n>>v;vector<int> Weight(n1);vector<int> Value(n1);vector<i…

PDF加密了无法编辑?解密方法来了!

一下午都在捣鼓各种格式问题&#xff0c;首先是需要合并几个 PDF&#xff0c;然而有一个文件加密了无法操作&#xff0c;碰到加密不能编辑就很头痛&#xff0c;终于让我找到一个可行的方法了&#xff0c; 首先就这个加密文件右键选择打开方式-Google Chrome>>打开>>…

环形链表——java

给你一个链表的头节点 head &#xff0c;判断链表中是否有环。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&#xff0c;评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置&#xff08;…

2024年Q1季度洗衣机行业线上市场销售数据分析

Q1季度洗衣机线上市场表现不如预期。 根据鲸参谋数据显示&#xff0c;2024年1月至3月线上电商平台&#xff08;京东天猫淘宝&#xff09;洗衣机累计销量约650万件&#xff0c;环比下降14%&#xff0c;同比下降14%&#xff1b;累计销售额约96亿元&#xff0c;环比下降30%&#…
最新文章