详解 Kruskal 算法的实现

一、算法原理

Kruskal 算法用于求最小生成树,它的主要思路是基于并查集,算法的主要原理如下:

假设图中有 n 个点,则:

  • step 1:Kruskal 算法假定初始时每个点都只属于自己所在的并查集(即初始时每个点都只属于一个独立的集合,各集合之间没有任何交集)。代码实现如下:

    // p[a] = S表示点a属于集合S
    for (int i = 1; i <= n; i++) p[i] = i;
    
  • step 2:将所有的边(假设图中共有 m 条)按权重从小到大进行排序(因为 Kruskal 算法是按照边的权重从小到大进行遍历的,排序的目的是为了保证在向当前的生成树中加入边时,加入的始终都是权重最小的边(其实也有点类似于贪心),这样当所有的边遍历完后,最终生成树中的所有边之和必然是全局最小的)。代码实现如下:

    // 重载Edge对象的"<",使之能够进行小于运算
    struct Edge
    {int a, b, w;bool operator<(const Edge& e) const{return w < e.w;}
    }edges[M];sort(edges, edges + m);
    
  • step 3:按边的权重,从小到大遍历所有边,取出边的两个顶点,然后进行判断:
    如果边的两个顶点不在同一个并查集中,则将这两个顶点加入同一个并查集(相当于将这两个顶点和它们之间的连边加入最小生成树),然后将边的长度累加至最终结果,并记录当前生成树中又加入了一条边(这个记录是为了判断图中有没有可能不存在最小生成树,如果最终得到的生成树中的边数不到 n − 1 n-1 n1 条( n n n 个点总共需有 n − 1 n-1 n1 条边),则说明图中不存在最小生成树;如果最终得到的边数等于 n − 1 n-1 n1 条,则说明图中存在最小生成树)。代码实现如下:

    int kruskal()
    {int res = 0, cnt = 0; // res是最终最小生成树的长度,cnt是全部循环结束以后所得生成树的边数for (int i = 0; i < m; i++) // 图中总共有m条边{auto e = edges[i]; // 先把这条边取出来int a = e.a, b = e.b, w = e.w; // 找到边的两个顶点和边的权重// 看这两个顶点是不是在同一个并查集中if (find(a) != find(b)) // 如果不在{p[find(a)] = find(b); // 把a所在的并查集加入b所在的并查集(相当于将两个并查集合并)res += w; // 累加上当前这条边的权重cnt++; // 记录当前生成树中又多了一条边}}// 根据最终cnt的取值判断图中到底存不存在最小生成树if (cnt < n - 1) return INF;else return res;
    }
    

二、完整代码实现

#include <iostream>
#include <cstring>
#include <algorithm> // sort()函数using namespace std;const int N = 1e5 + 10, M = 2e5 + 10, INF = 0x3f3f3f3f;struct Edge
{int a, b, w;bool operator<(const Edge& e) const{return w < e.w;}
}edges[M];int n, m;
int p[N];int find(int x)
{if (p[x] != x) p[x] = find(p[x]);return p[x];
}int kruskal()
{for (int i = 1; i <= n; i++) p[i] = i;sort(edges, edges + m);int res = 0, cnt = 0;for (int i = 0; i < m; i++){auto e = edges[i];int a = e.a, b = e.b, w = e.w;if (find(a) != find(b)){p[find(a)] = find(b);res += w;cnt++;}}if (cnt < n - 1) return INF;else return res;
}int main()
{cin >> n >> m;for (int i = 0; i < m; i++){int a, b, w;scanf("%d%d%d", &a, &b, &w);edges[i] = {a, b, w};}int res = kruskal();if (res == INF) puts("impossible");else cout << res << endl;return 0;
}

三、算法分析

  • Kruskal 算法的主要时间瓶颈在于对全部 m m m 条边的排序操作,故该算法的时间复杂度为 O ( m log ⁡ m ) O(m\log m) O(mlogm)
  • Kruskal 算法主要用于求 稀疏图 的最小生成树问题( m < n 2 m<n^2 m<n2)。

【注】以上内容参考 AcWing。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/661468.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

有趣的css - 简约的动态关注按钮

页面效果 此效果主要使用 css 伪选择器配合 css content 属性&#xff0c;以及 transition(过渡)属性来实现一个简约的动态按钮效果。 此效果可适用于关注按钮、详情按钮等&#xff0c;增强用户交互体验。 核心代码部分&#xff0c;简要说明了写法思路&#xff0c;看 css 部分的…

AUTOSAR内存篇 -Flash EEPROM Emulation(FEE)

文章目录 简介功能介绍通常行为寻址机制和分段地址计算擦除循环次数的限制“立即”数据的处理管理块正确性信息缓存对齐API介绍类型定义函数定义Fee_InitFee_ReadFee_WriteFee_CancelFee_GetStatus<

一文详解docker swarm

文章目录 1、简介1.1、涉及到哪些概念&#xff1f;1.2、需要注意什么&#xff1f; 2、集群管理2.1、创建集群2.2、将节点加入集群2.3、查看集群状态。2.4、将节点从集群中移除2.5、更新集群2.6、锁定/解锁集群 3、节点管理4、服务部署4.1、准备4.2、服务管理4.2.1、常用命令4.2…

[C++]继承(续)

一、基类和派生类对象赋值转换 在public继承时&#xff0c;父类和子类是一个“is - a”的关系。 子类对象赋值给父类对象/父类指针/父类引用&#xff0c;我们认为是天然的&#xff0c;中间不产生临时对象&#xff0c;也叫作父子类赋值兼容规则&#xff08;切割/切片&#xff…

idea查看日志的辅助插件 --- Grep Console (高亮、取消高亮)

&#x1f680; 分享一款很有用的插件&#xff1a;Grep Console &#x1f680; 我们在查看日志的时候可能会有遗漏&#xff0c;使用这款插件可以让特定的关键词高亮&#xff0c;可以达到不遗漏的效果&#xff01; 如果你是一个开发者或者对日志文件分析感兴趣&#xff0c;不要…

记录一次使用ant design 中 ConfigProvider来修改样式导致样式改变的问题(Tabs嵌套Tabs)

一 说明 继之前的一篇文章&#xff1a;antd5 Tabs 标签头的文本颜色和背景颜色修改 后&#xff0c;发现在被修改后的Tab中继续嵌套Tabs组件&#xff0c;这个新的Tabs组件样式跟外层Tabs样式也是一致的&#xff0c;如下图所示&#xff1a; 二 原因 在修改外层tabs样式时&…

Axios 和 Ajax 的区别

一、Axios 和 Ajax 的区别 1、Axios是一个基于Promise的HTTP库&#xff0c;而Ajax是对原生XHR的封装&#xff1b; 2、Ajax技术实现了局部数据的刷新&#xff0c;而Axios实现了对ajax的封装。 二、Axios 和 Ajax 的区别及优缺点 1、什么是Ajax Ajax是对原生XHR的封装&#xff0…

又涨又跌 近期现货黄金价格波动怎么看?

踏入2024年一月的下旬&#xff0c;现货黄金价格可以说没了之前火热的状态&#xff0c;盘面上是又涨又跌。面对这样的行情&#xff0c;很多投资者不知道如何看了。下面我们就来讨论一下怎么把握近期的行情。 先区分走势类型。在现货黄金市场中有两种主要的走势类型&#xff0c;一…

【SpringCloud】使用OpenFeign进行微服务化改造

目录 一、需求与背景二、OpenFeign 远程调用技术原理三、项目代码演示3.1 引入依赖3.2 实现OpenFeign注解修饰接口3.3 指定 OpenFeign 远程调用接口的扫描路径 四、OpenFeign 在日志中打印Request和Response五、OpenFeign 客户端超时配置六、使用 OpenFeign 实现服务降级6.1 实…

《区块链简易速速上手小册》第10章:区块链的未来与趋势(2024 最新版)

文章目录 10.1 区块链的未来展望10.1.1 基础知识10.1.2 主要案例&#xff1a;区块链在金融领域的发展10.1.3 拓展案例 1&#xff1a;区块链在供应链管理中的应用10.1.4 拓展案例 2&#xff1a;区块链在身份管理和隐私保护中的应用 10.2 新兴技术与区块链的融合10.2.1 基础知识1…

【已解决】QT如何加载qss文件

翻阅我的博客&#xff0c;发现没有写这方面文章&#xff0c;qt如何加载qss文件。今天补上一篇关于qt如何加载qss文件。写完这篇博客&#xff0c;未来有相关问题&#xff0c;直接可以从博客里去搜。 解决方案 在项目里写完qss文件&#xff0c;在main文件里只需要追加这句话。 …

智能家居的网关新形态:Aqara 方舟智慧中枢 M3 体验

如果说在刚刚结束的 2023 年有哪些备受期待的智能家居产品&#xff0c;Aqara 方舟智慧中枢 M3 一定榜上有名&#xff0c;我的多位朋友也曾在装修过程中多次向我询问是否有这款产品的相关资讯&#xff1b;谁能想到自从在 2022 年 11 月首次亮相之后&#xff0c;这款产品一直等了…

vulhub靶机activemq环境下的CVE-2015-5254(ActiveMQ 反序列化漏洞)

影响范围 Apache ActiveMQ 5.x ~ Apache ActiveMQ 5.13.0 远程攻击者可以制作一个特殊的序列化 Java 消息服务 (JMS) ObjectMessage 对象&#xff0c;利用该漏洞执行任意代码。 漏洞搭建 没有特殊要求&#xff0c;请看 (3条消息) vulhub搭建方法_himobrinehacken的博客-CSD…

iOS图像处理----探索图片解压缩到渲染的全过程以及屏幕卡顿

一&#xff1a;图像成像过程 ①、将需要显示的图像&#xff0c;由CPU和GPU通过总线连接起来&#xff0c;在CPU中输出的位图经总线在合适的时机上传给GPU &#xff0c;GPU拿到位图做相应位图的图层渲染、纹理合成。 ②、将渲染后的结果&#xff0c;存储到帧缓存区&#xff0c;帧…

堆解决贪心问题

502题&#xff0c;最小代价实现最大化IPO class Solution {/*本质还是贪心&#xff0c;只不过多了需要满足贪心的条件而已&#xff0c;首先由于启动资金是固定的&#xff0c;你得从小往大去观察哪些最小资本合适&#xff0c;这就需要定义一个pair&#xff0c;最小资本最大利润…

centos7常用命令之安装插件2

centos7安装插件1 7、kibana 【启动kibana,需要调整这个配置文件(/opt/kibana-6.3.0/config/kibana.yml)的一处ip地址,因为每次虚拟机的ip地址可能会有所不同&#xff0c; 同时访问页面地址的ip:5601时,ip地址也对应修改】 1.解压缩包 cd /opt/ tar -xvf kibana-6.3.0-linux-x…

【脑电信号处理与特征提取】P7-贾会宾:基于EEG/MEG信号的大尺度脑功能网络分析

基于EEG/MEG信号的大尺度脑功能网络分析 Q: 什么是基于EEG/MEG信号的大尺度脑功能网络分析&#xff1f; A: 基于脑电图&#xff08;EEG&#xff09;或脑磁图&#xff08;MEG&#xff09;信号的大尺度脑功能网络分析是一种研究大脑活动的方法&#xff0c;旨在探索脑区之间的功能…

【JavaSE篇】——继承

目录 &#x1f393;继承 ✅为什么需要继承 ✅继承概念 ✅继承的语法 ✅父类成员访问 &#x1f6a9;子类中访问父类的成员变量 1. 子类和父类不存在同名成员变量的情况 2. 子类和父类成员变量同名 &#x1f6a9;子类中访问父类的成员方法 1. 成员方法名字不同 2. 成员…

SAM:基于 prompt 的通用图像分割模型

Paper: Kirillov A, Mintun E, Ravi N, et al. Segment anything[J]. arXiv preprint arXiv:2304.02643, 2023. Introduction: https://segment-anything.com/ Code: https://github.com/facebookresearch/segment-anything SAM 是 Meta AI 开发的一款基于 prompt 的通用视觉大…

100%涨点!2024最新卷积块创新方案盘点(附模块和代码)

在写论文时&#xff0c;设计高效、创新的卷积块可以显著提升模型的性能&#xff0c;保障工作的有效性和先进性。另外&#xff0c;合理利用卷积块还可以帮助我们提升实验结果、拓展研究的视野和应用场景&#xff0c;增加论文的创新点。因此&#xff0c;对于论文er来说&#xff0…