并查集

并查集

并查集(Union-find Sets):是一种非常精巧而实用的数据结构,它主要用于处理一些不相交集合的合并问题。一些常见的用途有求连通子图、求最小生成树的 Kruskal 算法和求最近公共祖先(Least Common Ancestors, LCA)等。

     使用并查集时,首先会存在一组不相交的动态集合 S={
S1,S2,,Sk}
S={S1,S2,⋯,Sk},一般都会使用一个整数表示集合中的一个元素。

每个集合可能包含一个或多个元素,并选出集合中的某个元素作为代表。每个集合中具体包含了哪些元素是不关心的,具体选择哪个元素作为代表一般也是不关心的。我们关心的是,对于给定的元素,可以很快的找到这个元素所在的集合(的代表),以及合并两个元素所在的集合,而且这些操作的时间复杂度都是常数级的。

并查集的基本操作有三个:

      1.makeSet(s):建立一个新的并查集,其中包含 s 个单元素集合。

      2.unionSet(x, y):把元素 x 和元素 y 所在的集合合并,要求 x 和 y 所在的集合不相交,如果相交则不合并。

      3.find(x):找到元素 x 所在的集合的代表,该操作也可以用于判断两个元素是否位于同一个集合,只要将它们各自的代表比较一下就可以了。

       -对于并查集来说,每个集合用一棵树表示。
    -集合中每个元素的元素名分别存放在树的结点中,此外,树的每一个结点还有一个指向其双亲结点的指针。   
       -为简化讨论,忽略实际的集合名,仅用表示集合的树的根来标识集合。

下面给出两种路径压缩方法:

递归式路径压缩:

const int MAXSIZE = 500010;
int rank[MAXSIZE];    // 节点高度的上界
int parent[MAXSIZE]; // 根节点
int FindSet(int x){
   // 查找+递归的路径压缩
    if( x != parent[x] ) parent[x] = FindSet(parent[x]);
     return parent[x];
}
void Union(int root1, int root2){
     int x = FindSet(root1), y = FindSet(root2);
     if( x == y ) return ;
     if( rank[x] > rank[y] ) parent[y] = x;
     else{
         parent[x] = y;
         if( rank[x] == rank[y] ) ++rank[y];
     }
}
void Initi(void){
     memset(rank, 0, sizeof(rank));
     for( int i=0; i < MAXSIZE; ++i ) parent[i] = i;
}

非递归式路径压缩:

const int MAXSIZE = 30001;
int pre[MAXSIZE]; //根节点i,pre[i] = -num,其中num是该树的节点数目;
                   //非根节点j,pre[j] = k,其中k是j的父节点
int Find(int x){
   //查找+非递归的路径压缩
     int p = x;
     while( pre[p] > 0 )    p = pre[p];
     while( x != p ){
         int temp = pre[x]; pre[x] = p; x = temp;
     }
     return x;
}
void Union(int r1, int r2){
     int a = Find(r1); int b = Find(r2);
     if( a == b ) return ;
     //加权规则合并
     if( pre[a] < pre[b] ){
         pre[a] += pre[b]; pre[b] = a;
     }
     else {
         pre[b] += pre[a]; pre[a] = b;
     }
}
void Initi(void)
{
    for( int i=0; i < N; ++i ) pre[i] = -1;
}          

 

转载于:https://www.cnblogs.com/darklights/p/5302659.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/109115.html原文链接:https://javaforall.net

(0)
上一篇 2021年9月13日 上午9:00
下一篇 2021年9月13日 上午10:00


相关推荐

  • 【PotPlayer】敲好用的本地视频播放器

    【PotPlayer】敲好用的本地视频播放器软件简介PotPlayer是KMPlayer的原作者姜勇囍进入Daum公司后的新一代作品,目前仍有更新。由于采用Delphi编译程序的KMPlayer有一些弊端,姜勇囍为改进播放器本身的一些性能而重新用VC++进行构架。虽然PotPlayer与KMPlayer同属一个开发者的产品,但它与KMPlayer所注重的地方并不同,能够满足不同用户的使用需求。因该软件的官方网站托管于DAUM平台,中国大陆网络可能受防火长城(GFW)影响而无法正常访问。官方网站https://potplayer.daum.

    2022年7月14日
    43
  • 安装mysql 10055_MYSQL无法连接 提示10055错误的解决方法

    安装mysql 10055_MYSQL无法连接 提示10055错误的解决方法解决方法 以下内容为本人亲自实践原创 总结一下 应该是连接数的问题 那么服务器上有些什么连接数 1 IIS 网站服务器中各个网站中有 连接超时时间 会话超时时间 2 其它程序占用的服务器连接数 如 SMTP 服务在发信出去的时候可能有很多个连接数 3 服务器本身的 TCP IP 连接数 如 xp 系统就有个限制 不过 server2003 系统似乎没这个限制 解决操作 1 我的服务器上面有几个网站 其中有

    2026年3月17日
    2
  • 从“卖API”到“卖解决方案”,月之暗面Kimi押注Agent

    从“卖API”到“卖解决方案”,月之暗面Kimi押注Agent

    2026年3月12日
    2
  • DCOM配置

    DCOM配置要进行 DCOM 安全配置 操作者通常必须拥有客户和服务器计算机的管理员权限 nbsp 1 nbsp nbsp nbsp 用户的建立及配置 nbsp nbsp nbsp nbsp nbsp nbsp nbsp 最简单的用户配置是在客户和服务器计算机上建立名称 密码都相同的用户 Administrato 权限不是必需的 并用此用户登录系统 运行 OPC 服务器程序 这种方式适用于系统调试期间 或对安全要求不高的场合 nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp 在有一定安全要求的系统中 可按如下

    2026年3月18日
    2
  • SpringMVC 执行流程

    SpringMVC 执行流程springMVC(javaweb开发框架)1、MVC三层架构:模型(service、dao)、视图(JSP等)、控制器(Controller)什么是mvc?*MVC是模型、视图、控制器的简写,是一种软件设计规范*是将业务逻辑、数据、显示分离的方法来组织代码*MVC主要的作用就是降低了控制器(Controller)和视图(View)之间的双向耦合度*MVC不是一种设计模式、MVC是一种架构模式。当然不同的MVC存在着差异Model(数据模型):提供要展示的数据。因此包含数据和

    2022年6月28日
    27
  • 一元线性回归方程公式_用普通最小二乘法估计经典线性模型

    一元线性回归方程公式_用普通最小二乘法估计经典线性模型概述别看公式多,其实很简单最小二乘法其实又叫最小平方法,是一种数据拟合的优化技术。实质上是利用最小误差的平方寻求数据的最佳匹配函数,利用最小二乘法可以便捷的求得未知的数据,起到预测的作用,并且是的这些预测的数据与实际数据之间的误差平方和达到最小。一般应用在曲线拟合的目的上。原理本篇文章不考虑其他方面的应用,我们用最简单的实例说明最小二乘法的工作原理与其内在含义。当我们在研究两个…

    2025年6月1日
    4

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号