首页 > 编程语言 >「图::连通」详解并查集并实现对应的功能 / 手撕数据结构(C++)

「图::连通」详解并查集并实现对应的功能 / 手撕数据结构(C++)

时间:2024-10-19 17:47:53浏览次数:3  
标签:pre int 查集 C++ 集合 数据结构 root 节点 size

目录

概述

成员变量

创建销毁

根节点访问

路径压缩

启发式合并

复杂度

Code


概述

并查集,故名思议,能合并、能查询的集合,在图的连通性问题和许多算法优化上着广泛的使用。

这是一个什么数据结构呢?

一般来讲,并查集是由一系列集合组成的集合群。

其中,每个集合都有一个根节点,它的父亲仍是它自己,集合内其余的节点的父亲or祖宗均是这个节点,这样,一个根节点就领导了一个集合。而并查集支持这样的多个集合的访问以及合并操作。

每个集合都是一个树形结构,你可以认为并查集是一片森林。

听起来挺复杂度,但是其实很好实现。

成员变量

static constexpr int N = 1e5 + 5;一个无关紧要的常量,限制最大值。

int pre[N];父节点指针数组,pre[i]=j表示i的父亲是j,即二者处于同一集合。

int size[N];集合大小数组,只用每个集合的根节点上的size是有意义的,若i为根节点size[i]=x表示i所在的集合大小为x。

struct union_find_set {
	static constexpr int N = 1e5 + 5;
	int pre[N],size[N];
    ...
};

创建销毁

在一开始,每个节点都单独成为一个集合,每个集合大小为1。

union_find_set(int n = N) {
	for (int i = 0; i < n; i++) {
		pre[i] = i;
		size[i] = 1;
	}
}

根节点访问

给定一个节点x,我们想访问他的根节点

这一步可以递归实现:我们知道只有根节点的pre指向自己,所以如果pre[x]==x,我们就找到了根节点,否则沿着pre指针爬,传入下一级root。

int root(int x) {
	return pre[x] == x ? x : root(pre[x]);
}

路径压缩

沿着pre指针爬树的过程的时间复杂度是线性级别的,但是我们可以使用路径压缩

当我们依次跳出递归时,额外将这一级x的pre指针更新为找到的根节点,这样,一棵树就由多层被压缩成了两层(根节点与叶子节点)。

int root(int x) {
	return pre[x] = (pre[x] == x ? x : root(pre[x]));
}

启发式合并

想合并两个集合,我们可以采用启发式合并,即按规模合并。

路径压缩并不是实时的,所以一般一个集合仍是多层的。在这种情况下,为了保持查询根节点的高效性,我们应该将小集合接在大集合之下(小集合中节点少,向上访问的总代价小,如果反过来,那么大集合中的大量节点想向上访问会极其不利)。

给出两个节点,将其所在的集合合并,我们先找出root,如果两个节点确实属于不同集合,我们将小集合接在大集合之下,这样就能保持根节点查询的效率。最后根节点的size。

void unite(int x, int y) {
	x = root(x), y = root(y);
	if (x == y)return;
	if (size[y] < size[x])std::swap(x, y);
	pre[x] = y;
	size[y] += size[x];
}

复杂度

时间复杂度:O(n) 使用路径压缩:O(1)~O(logn)

空间复杂度:O(n) 

Code

#include <algorithm>
#ifndef UNION_FIND_SET
#define UNION_FIND_SET
struct union_find_set {
	static constexpr int N = 1e5 + 5;
	int pre[N],size[N];
	union_find_set(int n) {
		for (int i = 0; i < n; i++) {
			pre[i] = i;
			size[i] = 1;
		}
	}
	int root(int x) {
		return pre[x] = (pre[x] == x ? x : root(pre[x]));
	}
	int get_size(int x) {
		return size[root(x)];
	}
	void unite(int x, int y) {
		x = root(x), y = root(y);
		if (x == y)return;
		if (size[y] < size[x])std::swap(x, y);
		pre[x] = y;
		size[y] += size[x];
	}
};
#endif

标签:pre,int,查集,C++,集合,数据结构,root,节点,size
From: https://blog.csdn.net/dakingffo/article/details/143081327

相关文章

  • 学习记录,这该死的c++
    最近还是比较懈怠,除了老师布置的作业其他的也是匆匆过一眼至于代码方面最主要的问题还是不理解该怎么表达。总的来说还是要在多沉淀。为什么会有知道敲代码但是不知道该怎么成功表达这个问题啊?还是不能把代码敲得精简一些可以来个人教我怎么敲关系符号吗?运用的还不是很熟练......
  • 【C++贪心】2086. 喂食仓鼠的最小食物桶数|1622
    本文涉及知识点C++贪心LeetCode2086.喂食仓鼠的最小食物桶数给你一个下标从0开始的字符串hamsters,其中hamsters[i]要么是:‘H’表示有一个仓鼠在下标i,或者’.’表示下标i是空的。你将要在空的位置上添加一定数量的食物桶来喂养仓鼠。如果仓鼠的左边或右边......
  • 南沙C++信奥赛陈老师解一本通题 1286:怪盗基德的滑翔翼
    ​【题目描述】怪盗基德是一个充满传奇色彩的怪盗,专门以珠宝为目标的超级盗窃犯。而他最为突出的地方,就是他每次都能逃脱中村警部的重重围堵,而这也很大程度上是多亏了他随身携带的便于操作的滑翔翼。有一天,怪盗基德像往常一样偷走了一颗珍贵的钻石,不料却被柯南小朋友识破了伪......
  • C++之子类继承与父类构造
    C++之子类继承与父类构造文章目录C++之子类继承与父类构造1.问题引出2.原则3.解析3.1单一继承3.1.1父类无参构造函数3.1.1.1子类无定义构造函数3.1.1.2子类定义构造函数3.1.2父类存在无参构造函数和有参构造函数3.1.3父类仅存在有参构造函数3.2链式继承3.3多继承......
  • 【C++】C++中的继承,看这一篇就够了
    【C++】C++中的继承,看这一篇就够了一.继承的概念及定义继承的概念继承定义继承关系和访问限定符继承基类成员访问方式的变化二.基类和派生类对象赋值转换三.继承中的作用域四.派生类的默认成员函数五.继承与友元六.继承与静态成员七.复杂的菱形继承及菱形虚拟继承......
  • 数据结构与算法之线性表的基本操作
    数据结构之线性表的基本操作初始化,插入,获取#include<stdio.h>#include<stdlib.h>#include<malloc.h>#defineOK1#defineOVERFLOW-1#defineLIST_INIT_SIZE100#defineLISTINCREMENT10typedefintElemType;typedefstruct{ ElemType*elem; intlength; i......
  • 移情别恋c++ ദ്ദി˶ー̀֊ー́ ) ——13.map&&set(无习题)
    C++中的set和map容器详细总结1.概述C++标准模板库(STL)提供了多种关联容器,用于管理键值对和集合的数据。其中,set和map是最常用的两种关联容器。set用于存储唯一的元素集合,而map则用于存储键值对,其中每个键都是唯一的。它们都使用红黑树(自平衡二叉搜索树)作为底......
  • YU_C++算法学习笔记 · 枚举
    1.1枚举类问题·枚举是什么?枚举也叫穷举,是计算机解决问题最基本的策略。其方法是一一列举所有的可能性,根据题意要求进行合理的判断或计算,最终得到答案,本质上就是一种搜索算法基础的枚举就是人们常说的“暴力”求解。对于不同的问题,不可过分依赖“暴力”求解,应该根据具体的......
  • 《 C++ 修炼全景指南:十六 》玩转 C++ 特殊类:C++ 六种必备特殊类设计的全面解析
    摘要这篇博客深入探讨了六种C++特殊类的设计及其技术细节。首先,介绍了如何设计只能在堆上或栈上创建对象的类,通过控制构造函数的访问权限来限定对象的内存分配区域。接着,探讨了如何设计一个不能被拷贝的类,避免资源重复释放的问题。随后,介绍了如何防止类被继承以及单例模......
  • 位置式与增量式PID控制器理论与C++实现
    1理论推导1.1PID式中: ——控制器的输出;——控制器的输入(常常是设定值与被控量之差);Kp——控制器的比例放大系数;Ti——控制器的积分时间;Td——控制器的微分时间;1.2位置式PID设为第k次采样时刻控制器的输出值,可得离散的PID算式,又称位置式PID算式:e(k):用户设定的值(目标......