并查集

并查集

并查集(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


相关推荐

  • jenkins拉取gitlab代码_git提交远程仓库命令

    jenkins拉取gitlab代码_git提交远程仓库命令前言python自动化的脚本开发完成后需提交到git代码仓库,接下来就是用Jenkins拉取代码去构建自动化代码了新建项目打开Jenkins新建一个自由风格的项目源码管理Repository

    2022年7月29日
    10
  • JSP源代码大全–倒计时代码生成器

    JSP源代码大全–倒计时代码生成器请参照以下的原代码 注意 可以替换 JavaScript 码中的 2007 以下是网页源代码 1 将第一部分粘贴到 HTML 的 HEAD 区 2 将 OnLoad 事件加入 BODY 标签内 3 将最后一部分代码加入 BODY 区 varTemp2 vartimerID null vartimerRunn false functionarry this length

    2026年3月26日
    2
  • pycharm加注释的快捷方式_pycharm全部注释

    pycharm加注释的快捷方式_pycharm全部注释Ctrl+F1显示错误描述或警告信息Alt+Enter快速修正Ctrl+R替换Ctrl+Shift+F或者连续2次敲击shift全局查找{可以在整个项目中查找某个字符串什么的,如查找某个函数名字符串看之前是怎么使用这个函数的}Ctrl+Shift+R全局替换Alt+Shift+F10运行模式配置Alt+…

    2022年8月28日
    8
  • siamfc-pytorch代码讲解(三):demo&track

    siamfc-pytorch代码讲解(三):demo&track我之前的两篇博客:siamfc-pytorch代码讲解(一):backbone&headsiamfc-pytorch代码讲解(二):train&siamfc代码来自:https://github.com/huanglianghua/siamfc-pytorch今天主要看一下demo的部分,也就是涉及到测试tracking的部分。直接上代码:一、demo.pyfro…

    2022年10月1日
    4
  • matlab marker太多,关于plot中的markersize问题

    matlab marker太多,关于plot中的markersize问题最近用 Plot 在同一个图中画出多条曲线 为示区别每条曲线 这里使用 等 marker 但是画出的图却是一整片 好像是相邻两个 marker 的间距太近的缘故 显示不出这里的 marker 所以我想问下 能否将相邻的 marker 间距调大些 以显示出各条曲线的 marker 来 我的程序为 clcclearem 3 5e 11 t 242 h 221 w 249 s 105 5 1000 s1 249 w1

    2026年3月17日
    2
  • PTA 列车调度 python

    PTA 列车调度 python火车调度PTApython实现两端分别是一条入口(Entrance)轨道和一条出口(Exit)轨道,它们之间有N条平行的轨道。每趟列车从入口可以选择任意一条轨道进入,最后从出口离开。在图中有9趟列车,在入口处按照{8,4,2,5,3,9,1,6,7}的顺序排队等待进入。如果要求它们必须按序号递减的顺序从出口离开,则至少需要多少条平行铁轨用于调度?

    2022年7月14日
    20

发表回复

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

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