最小生成树(MST)的Kruskal算法

内容来源 紫皮书p356 kruskal算法的思想在于合并


Kruskal算法的第一步是给所有边按照从小到大的顺序排列。这一步可以直接使用sort。接下来从小到大依次考察每条边(u,v)。
u,v如果在同一个连通分量中,那么加入(u,v)后会成环

伪代码:

把所有边排序,记第i小的边为e[i];
初始化MST为空
初始化连通分量,让每个点自成一个独立的连通分量
for(int i = 0; i < m; i++)
	if(e[i].u和e[i].v不在同一个连通分量) {
   
			把边e[i]加入MST
			合并e[i].u和e[i].v所在的连通分量
	}

最关键的在于连通分量的查询和合并
需要知道任意两个点是否在同一个连通分量中
还需要合并两个分量
因此要使用并查集,把每个连通分量看成一个集合,每个点恰好属于一个连通分量,图的所有连通分量可以用若干不相交集合表示

Kruskal的完整代码
假设第i条边的两个端点序号和权值分别储存在u[i],v[i],w[i]中

int cmp(const int i, const int j) {
   return w[i]<w[j]; }//间接排序
int find(int x) {
   return p[x] == x ? x : p[x] = find(p[x]);}
int Kruskal() {
   
	int ans = 0;
	for(int i = 0; i < n; i++) p[i] = i;//初始化并查集
	for(int i = 0; i < m; i++) r[i] = i;//初始化边序号
	sort(r,r+m,cmp);
	for(int i = 0; i < m; i++) {
   
		int e = r[i]; int x = find(u[e]); int y = find(v[e]);
		if(x != y) {
   ans += w[e]; p[x] = y;}
	}
	return ans;
}
全部评论

相关推荐

拒绝996的悲伤蛙:此贴终结|给路过的牛友分享一下心得👇 实习的时候不要光埋头干活,身边的大佬同事才是真·宝藏人脉!大胆请教他们工作以及职场上的问题以我的经历,我的带教有十几年工作经验,做过运维、后端开发、web测试,现在是高级软测,是行走的避坑指南 我之前纠结要不要学Web测试简历,被他一句话点醒:Web发展成熟,岗位需求在缩,AI对互联网的冲击可能以后架构+开发+测试一人包揽。现在用户更多用的是移动端APP/小程序,相比之下天天守着电脑刷网页的人基数小。 这里我的纠结得到反馈,于是我又把简历发给带教,获得了一对一的简历指导。 感兴趣的可以看看: 1.教育背景:本科→本科(全日制) 2.实习经历:总体问题不大,但第2点要稍作修改,可以写但做功课,如风机、水箱……可能会问用哪个供应商的?使用寿命、型号、电压电流、多少秒会触发逻辑? 3.项目经历(坑太多,大型翻车现场): - 项目名越直白越好,让人一眼就知道你干了啥。 -用的什么语言设计核心接口,异步执行做功课,涉及线程问题,被问可回答n个功能是如何错开异步执行的 - “验证任务消费……阻塞丢包”“高负载稳定性”这种词,没三五年开发功底不要写,不然面试时被问线程、数量级、CPU占用,内存带宽等影响性能的直接原地社死。 -做功课 -做功课,测了哪些模块,如何设计,接口流量抓包,token,变量…… -做功课,要熟悉网络协议…… 带教之前做互联网开发的时候面试过很多人,总的来说不要为了显得项目高大上过渡包装,写了就要做好拷打的准备
听劝,我这个简历该怎么改...
点赞 评论 收藏
分享
评论
1
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务