热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

数据库技术:并查集

publicclassUnionSet{publicstaticclassNode{Vvalue;publicNode(Vvalue){this.value

并查集的定义

并查集:一种维护集合的数据结构,合并(Union),查找(Find),集合(Set)

  1. 合并:合并两个集合
  2. 查找:判断两个元素是否在一个集合
    并查集实现——用一个数组
int father[N}; 

father[i]表示元素i的父亲结点,而父亲结点也是这个集合的元素(1<=i<=N)。
father[i]=i:元素i是该集合的根结点
同一集合来说只存在一个根结点,且将其作为所属集合的标识
并查集

father[1] = 1; //1的父亲结点是自己,即根结点 father[2] = 1; //2的父亲结点是1 father[3] = 2; //3的父亲结点是2 father[4] = 2; //4的父亲结点是2 father[5] = 5; //5的父亲结点是自己,即根结点 father[6] = 6; //6的父亲结点是5 

并查集的基本操作

  1. 初始化
    每个元素都是独立的一个集合,需要令所有father[i]=i
for(int i = 1; i <= N; i++) {     father[i] = i; //令father[i]为-1也可,此处以father[i] = i为例 } 
  1. 查找
    对给定的结点寻找其根结点的过程
//findFather函数返回元素x所在集合的根结点 int findFather(int x) {     while(x != father[x]) //如果不是根结点,继续循环     {         x = father[x]; //获得自己的父亲结点     }     return x; } 

也可以用递归实现

int findFather(int x) {     if(x == father[x])         return x; //如果找到根结点,则返回根结点编号x     else         return findFather(father[x]); //否则,递归判断x的父亲结点是否是根结点 } 
  1. 合并
    把两个集合合并成一个集合。
    先判断两个元素是否属于同一个集合,只有当两个元素属于不同集合时合并,把其中一个集合的根结点的父亲指向另一个集合的根结点。
    1)对于给定的两个元素a、b,判断它们是否属于同一集合。调用上面的查找函数,对a、b分别查找根结点,然后再判断其根结点是否相同。
    2)合并两个集合:在1)中已经获得两个元素的根结点faA与faB,因此只需要把其中一个的父亲结点指向另一个结点。father[faA]=faB
    并查集
    1)判断元素4和元素6是否属于同一个集合:元素4所在集合的根结点是1,元素6所在集合的根结点是5,不属于同一集合
    2)合并两个集合:令father[5]=1,把元素5的父亲设为元素1
void Union(int a, int b) {     int faA = findFather(a); //查找a的根结点,记为faA     int faB = findFather(b); //查找b的根结点,记为faB     if(faA != faB) //如果不属于同一个集合     {         father[faA] = faB; //合并它们     } } 

并查集产生的每一个集合都是一棵树。

路径压缩

把当前查询结点的路径上的所有结点的父亲都指向根结点
并查集

  1. 按原来的写法获得x的根结点r
  2. 重新从x开始走一遍寻找根结点的过程,把路径上经过的所有结点的父亲全部改为根结点r
int findFather(int x) {     //由于x在下面的while中会变成根结点,因此先把原先的x保存一下     int a = x;     while(x != father[x]) //寻找根结点     {         x = father[x];     }     //到这里,x存放的是跟结点。下面把路径上所有结点的father都改成根结点     while(a != father[a])     {         int z = a; //因为a要被father[a]覆盖,所以先保存a的值,以修改father[a]         a = father[a]; //a回溯父亲结点         father[z] = x; //将原先的结点a的父亲改成根结点x     }     return x; //返回根结点 } 

递归写法

int findFather(int v) {     if(v == father[v])         return v; //找到根结点     else     {         int F = findFather(father[v]); //递归寻找father[v]的根结点F         father[v] = F; //将根结点F赋给father[v]         return F; //返回根结点F     } }  

需要了解更多数据库技术:并查集,都可以关注数据库技术分享栏目—编程笔记


推荐阅读
  • 本文介绍了C#中数据集DataSet对象的使用及相关方法详解,包括DataSet对象的概述、与数据关系对象的互联、Rows集合和Columns集合的组成,以及DataSet对象常用的方法之一——Merge方法的使用。通过本文的阅读,读者可以了解到DataSet对象在C#中的重要性和使用方法。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • 本文详细介绍了SQL日志收缩的方法,包括截断日志和删除不需要的旧日志记录。通过备份日志和使用DBCC SHRINKFILE命令可以实现日志的收缩。同时,还介绍了截断日志的原理和注意事项,包括不能截断事务日志的活动部分和MinLSN的确定方法。通过本文的方法,可以有效减小逻辑日志的大小,提高数据库的性能。 ... [详细]
  • 本文介绍了lua语言中闭包的特性及其在模式匹配、日期处理、编译和模块化等方面的应用。lua中的闭包是严格遵循词法定界的第一类值,函数可以作为变量自由传递,也可以作为参数传递给其他函数。这些特性使得lua语言具有极大的灵活性,为程序开发带来了便利。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • [译]技术公司十年经验的职场生涯回顾
    本文是一位在技术公司工作十年的职场人士对自己职业生涯的总结回顾。她的职业规划与众不同,令人深思又有趣。其中涉及到的内容有机器学习、创新创业以及引用了女性主义者在TED演讲中的部分讲义。文章表达了对职业生涯的愿望和希望,认为人类有能力不断改善自己。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • 给定一个二维平面上的一些点,通过计算曼哈顿距离,求连接所有点的最小总费用。只有任意两点之间有且仅有一条简单路径时,才认为所有点都已连接。给出了几个示例并给出了对应的输出。 ... [详细]
  • C# 7.0 新特性:基于Tuple的“多”返回值方法
    本文介绍了C# 7.0中基于Tuple的“多”返回值方法的使用。通过对C# 6.0及更早版本的做法进行回顾,提出了问题:如何使一个方法可返回多个返回值。然后详细介绍了C# 7.0中使用Tuple的写法,并给出了示例代码。最后,总结了该新特性的优点。 ... [详细]
  • 1,关于死锁的理解死锁,我们可以简单的理解为是两个线程同时使用同一资源,两个线程又得不到相应的资源而造成永无相互等待的情况。 2,模拟死锁背景介绍:我们创建一个朋友 ... [详细]
  • 后台获取视图对应的字符串
    1.帮助类后台获取视图对应的字符串publicclassViewHelper{将View输出为字符串(注:不会执行对应的ac ... [详细]
  • 动态规划算法的基本步骤及最长递增子序列问题详解
    本文详细介绍了动态规划算法的基本步骤,包括划分阶段、选择状态、决策和状态转移方程,并以最长递增子序列问题为例进行了详细解析。动态规划算法的有效性依赖于问题本身所具有的最优子结构性质和子问题重叠性质。通过将子问题的解保存在一个表中,在以后尽可能多地利用这些子问题的解,从而提高算法的效率。 ... [详细]
author-avatar
lovely月夜静悄悄知_302
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有