并查集及应用

/*********************************************************************************
数据结构——并查集及应用
并查集是若干个不相交集合,能够实现较快的合并和判断元素所在集合的操作,应用很多。
一般采取树形结构来存储并查集,并利用一个rank数组来存储集合的深度下界,在查找操作时进行路径压缩使后续的查找操作加速。
这样优化实现的并查集,空间复杂度为O(N),建立一个集合的时间复杂度为O(1),
N次合并M查找的时间复杂度为O(M Alpha(N)),这里Alpha是Ackerman函数的某个反函数,
在很大的范围内(人类目前观测到的宇宙范围估算有10的80次方个原子,这小于前面所说的范围)
这个函数的值可以看成是不大于4的,所以并查集的操作可以看作是线性的。
**********************************************************************************/

int   set [MAXN],rank[MAXN];

int  FindSet( int  x)
{
    
if(set[x]!=x)
        
set[x]=FindSet(set[x]);
    
return set[x];
}


void  MakeSet( int  x)
{
    
set[x]=x;
    rank[x]
=0;
}


void  Link( int  a, int  b)
{
    
if(rank[a]>rank[b])
        
set[b]=a;
    
else if(rank[a]<rank[b])
        
set[a]=b;
    
else
    
{
        
set[a]=b;
        rank[b]
++;
    }

}


void  Union( int  a, int  b)
{
    Link(FindSet(a),FindSet(b));
}


 


这是通过并查集实现kruskal算法求图的最小生成树的例子:
以下内容为程序代码:

#include  " stdio.h "
#include 
" string.h "
#define  INFIN 30000
#define  MAXN 10

int   set [MAXN],rank[MAXN];

int  FindSet( int  x)
{
    
if(set[x]!=x)
        
set[x]=FindSet(set[x]);
    
return set[x];
}


void  MakeSet( int  x)
{
    
set[x]=x;
    rank[x]
=0;
}


void  Link( int  a, int  b)
{
    
if(rank[a]>rank[b])
        
set[b]=a;
    
else if(rank[a]<rank[b])
        
set[a]=b;
    
else
    
{
        
set[a]=b;
        rank[b]
++;
    }

}



void  Union( int  a, int  b)
{
    Link(FindSet(a),FindSet(b));
}


 
int  main()
{
    
int n,k,i,j,map[MAXN][MAXN];
    
int t1,t2,t3,cost=0;
    memset(map,
0,sizeof(map));
    scanf(
"%d %d",&n,&k);
    
while(k--)
    
{
        scanf(
"%d %d %d",&t1,&t2,&t3);
        map[t1][t2]
=map[t2][t1]=t3;
    }

    
for(i=0;i<n;i++)
        MakeSet(i);
    
for(k=1;k<n;k++)
    
{
        t3
=INFIN;
        
for(i=0;i<n;i++)
            
for(j=0;j<n;j++)
                
if(map[i][j] && FindSet(i)!=FindSet(j))
                    
if(map[i][j]<t3)
                    
{
                        t1
=i;t2=j;t3=map[i][j];
                    }

        cost
+=t3;
        Union(t1,t2);
    }

    printf(
"%d/n",cost);
    
return 0;
}
 
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值