【数据结构】图的最小生成树



快乐的流畅:个人主页


个人专栏:《C游记》《进击的C++》《Linux迷航》

远方有一堆篝火,在为久候之人燃烧!

文章目录

  • 引言
  • 一、最小生成树的概念
  • 二、Kruskal算法
    • 2.1 思想
    • 2.2 实现
  • 三、Prim算法
    • 3.1 思想
    • 3.2 实现
  • 四、Kruskal和Prim的对比

引言

前置知识:【数据结构】并查集
前置知识:【数据结构】图的概念和存储结构

一、最小生成树的概念

最小生成树(Minimum Spanning Tree, MST) 是图论中的一个重要概念,指的是在一个连通的无向图中,选择一部分边,使得这些边连接所有顶点且边权值之和最小,同时生成的子图仍是一个树结构(没有环)。


按照定义,若连通网络由n个顶点组成,则其生成树必含n个顶点,n-1条边。因此,构造最小生成树有3个准则:

  1. 只能使用该网络的边来构造最小生成树
  2. 只能使用恰好n-1条边来连接n个顶点
  3. 选用的这n-1条边不能构成回路


下面讲解两种经典的最小生成树算法——kruskal算法和prim算法,它们都采用了贪心策略来进行实现。

前置声明:

template<class W>
struct Edge
{int _srci;int _dsti;W _w;Edge(int srci, int dsti, const W& w): _srci(srci), _dsti(dsti), _w(w){}bool operator>(const Edge& e) const{return _w > e._w;}
};template<class V, class W, W W_MAX = INT_MAX, bool Direction = false>
class Graph
{typedef Edge<W> Edge;typedef Graph<V, W, W_MAX, Direction> Self;//...
}

二、Kruskal算法

2.1 思想

Kruskal算法采用全局贪心的策略:

  1. 每次选取图中权值最小的边。
  2. 每次选取完后,判断是否构成回路。若构成,则舍弃这条边。

具体图示如下:

2.2 实现

思路:

  1. 采用优先级队列(小堆),将所有边存入,以便每次选取全图权值最小的边。
  2. 采用并查集,存储已选取的顶点。每次选取边时,判断两侧顶点是否在并查集中,以此判断是否构成回路。
W Kruskal(Self& minTree)
{minTree._vertexs = _vertexs;minTree._indexMap = _indexMap;int n = _edges.size();minTree._edges.resize(n, vector<W>(n, W_MAX));priority_queue<Edge, vector<Edge>, greater<Edge>> minHeap;for (int i = 0; i < n; ++i){for (int j = 0; j < n; ++j){if (i < j && _edges[i][j] != W_MAX)//无向图,防止重复记录路径{minHeap.push(Edge(i, j, _edges[i][j]));}}}UnionFindSet<V> ufs(n);W total = W();int count = 0;while (!minHeap.empty()){Edge top = minHeap.top();minHeap.pop();int srci = top._srci, dsti = top._dsti;W w = top._w;if (!ufs.InSet(srci, dsti)){minTree._AddEdge(srci, dsti, w);ufs.Union(srci, dsti);total += w;++count;if (count == n - 1){break;}cout << _vertexs[srci] << "--" << _vertexs[dsti] << ":" << w << endl;}else{cout << _vertexs[srci] << "--" << _vertexs[dsti] << ":" << w << "[构成环]" << endl;}}if (count == n - 1){return total;}else{return W();}
}

细节:

  1. 输入一个空图,输出最小生成树的总权值,并将空图变为最小生成树。
  2. count记录已选边数,若达到n-1,则可直接跳出循环,提高效率。

三、Prim算法

3.1 思想

Prim算法采用局部贪心的策略:

  1. 将已被选择的点看作一个顶点集合,初始状态只有起点在集合中。
  2. 每次在集合周围查找,找到消耗权值最小即可抵达的点,并将其纳入集合。

具体图示如下:

3.2 实现

思路:

  1. 采用优先级队列(小堆),将起始点周围的路径存入,以便每次选取集合周围权值最小的边。
  2. 集合用bool数组表示,每次只有当目标点不在集合中,才将其纳入集合。
  3. 每次将新点纳入集合后,再将新点周围的路径存入优先级队列,依次迭代。
W Prim(Self& minTree, const V& src)
{minTree._vertexs = _vertexs;minTree._indexMap = _indexMap;int n = _edges.size();minTree._edges.resize(n, vector<W>(n, W_MAX));//起始点周围的路径入堆priority_queue<Edge, vector<Edge>, greater<Edge>> minHeap;int srci = GetIndex(src);for (int i = 0; i < n; ++i){if (_edges[srci][i] != W_MAX){minHeap.push(Edge(srci, i, _edges[srci][i]));}}vector<bool> S(n, false);S[srci] = true;W total = W();int count = 0;while (!minHeap.empty()){Edge top = minHeap.top();minHeap.pop();int srci = top._srci, dsti = top._dsti;W w = top._w;if (!S[dsti]){minTree._AddEdge(srci, dsti, w);S[dsti] = true;total += w;++count;if (count == n - 1){break;}for (int i = 0; i < n; ++i){if (!S[i] && _edges[dsti][i] != W_MAX)//无向图,防止重复记录路径{minHeap.push(Edge(dsti, i, _edges[dsti][i]));}}cout << _vertexs[srci] << "--" << _vertexs[dsti] << ":" << w << endl;}else{cout << _vertexs[srci] << "--" << _vertexs[dsti] << ":" << w << "[构成环]" << endl;}}if (count == n - 1){return total;}else{return W();}
}

细节:

  1. 输入一个空图和一个起始点,输出最小生成树的总权值,并将空图变为最小生成树。
  2. count记录已选边数,若达到n-1,则可直接跳出循环,提高效率。

四、Kruskal和Prim的对比

Kruskal 算法Prim 算法
算法类型贪心算法贪心算法
适用图类型稀疏图,特别适合边权值分布较广的图稠密图,特别适合顶点多边多的图
基本思想从边的角度出发,按权值从小到大选择边从顶点出发,每次扩展最小权值的边
时间复杂度O(E log V)O(E log V)
典型应用网络设计问题,边多且分散的图电网、网络设计问题,稠密的图
贪心选择标准每次选择权值最小且不形成环的边每次选择最小的连接权值,扩展已加入的顶点集合

  • Kruskal:适用于稀疏图,因为其从边的角度出发,边的数量相对少时效率更高。
  • Prim:适用于稠密图,因为它每次从顶点出发,逐渐扩展树,对于稠密图(边多的图)效率更高

真诚点赞,手有余香

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

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

相关文章

Spring Task 调度任务

Spring Task是调度任务框架&#xff0c;通过配置&#xff0c;程序可以按照约定的时间自动执行代码逻辑&#xff0c;基于注解方式实现需要如下注解&#xff1a; Component 任务调度类交给Spring IOC容器管理EnableScheduling 启用 Spring 的定时任务&#xff08;Scheduling&…

索尼MDR-M1:超宽频的音频盛宴,打造沉浸式音乐体验

在音乐的世界里&#xff0c;每一次技术的突破都意味着全新的听觉体验。 索尼&#xff0c;作为音频技术的先锋&#xff0c;再次以其最新力作——MDR-M1封闭式监听耳机&#xff0c;引领了音乐界的新潮流。 这款耳机以其超宽频播放和卓越的隔音性能&#xff0c;为音乐爱好者和专…

k8s中,ingress的实现原理,及其架构。

图片来源&#xff1a;自己画的 图片来源&#xff1a;k8s官网 首先&#xff0c;什么是ingress? 是服务还是控制器&#xff1f; 都不精确 ingress是一个api资源 service和deployment也是api资源。 这几个相互协作&#xff0c;组建成一个对外提供服务的架构。 ingress提供的…

[C++]使用纯opencv部署yolov11目标检测onnx模型

yolov11官方框架&#xff1a;https://github.com/ultralytics/ultralytics 【算法介绍】 在C中使用纯OpenCV部署YOLOv11进行目标检测是一项具有挑战性的任务&#xff0c;因为YOLOv11通常是用PyTorch等深度学习框架实现的&#xff0c;而OpenCV本身并不直接支持加载和运行PyTor…

系统安全 - RedisMySQL安全及实践

文章目录 导图Redis 安全潜在的安全风险防护措施密码认证命令重命名权限最小化日志和审计 Red网络隔离 MySQL 安全认证和授权文件操作风险传输和存储加密最小权限原则审计 总结 导图 Redis 安全 Redis的设计初衷是为了在可信环境下提供高性能的KV数据库服务&#xff0c;因此它…

FiBiNET模型实现推荐算法

1. 项目简介 A031-FiBiNET模型项目是一个基于深度学习的推荐系统算法实现&#xff0c;旨在提升推荐系统的性能和精度。该项目的背景源于当今互联网平台中&#xff0c;推荐算法在电商、社交、内容分发等领域的广泛应用。推荐系统通过分析用户的历史行为和兴趣偏好&#xff0c;预…

【NIO基础】NIO(非阻塞 I/O)和 IO(传统 I/O)的区别,以及 NIO 的三大组件详解

目录 1、NIO 2、NIO 和 IO 的区别 1. 阻塞 vs 非阻塞 2. 一个线程 vs 多个连接 3. 面向流 vs 面向缓冲 4. 多路复用 3、Channel & Buffer (1&#xff09;Channel&#xff1a;双向通道 (2&#xff09;Buffer&#xff1a;缓冲区 (3&#xff09;ByteBuffer&#xff…

用Arduino单片机读取PCF8591模数转换器的模拟量并转化为数字输出

PCF8591是一款单芯片&#xff0c;单电源和低功耗8位CMOS数据采集设备。博文[1]对该产品已有介绍&#xff0c;此处不再赘述。但该博文是使用NVIDIA Jetson nano运行python读取输入PCF8591的模拟量的&#xff0c;读取的结果显示在屏幕上&#xff0c;或输出模拟量点亮灯。NVIDIA J…

计算机毕业设计 基于Python的智能文献管理系统的设计与实现 Python+Django+Vue 前后端分离 附源码 讲解 文档

&#x1f34a;作者&#xff1a;计算机编程-吉哥 &#x1f34a;简介&#xff1a;专业从事JavaWeb程序开发&#xff0c;微信小程序开发&#xff0c;定制化项目、 源码、代码讲解、文档撰写、ppt制作。做自己喜欢的事&#xff0c;生活就是快乐的。 &#x1f34a;心愿&#xff1a;点…

讯飞星火编排创建智能体学习(四):网页读取

目录 引言 网页读取节点 如何生成网址 测试 引言 在讯飞星火编排创建智能体学习&#xff08;三&#xff09;&#xff1a;搜索工具-CSDN博客中&#xff0c;我介绍了如何用搜索工具从网上搜索车次信息。不过&#xff0c;在测试中我们也发现讯飞星火的这个工具并不是特别完善&…

誉天Linux云计算课程学什么?为什么保障就业?

一个IT工程师相当于干了哪些职业? 其中置顶回答生动而形象地描绘道&#xff1a; 一个IT工程师宛如一个超级多面手&#xff0c;相当于——加班狂程序员测试工程师实施工程师网络工程师电工装卸工搬运工超人。 此中酸甜苦辣咸&#xff0c;相信很多小伙伴们都深有体会。除了典…

ESP01 AT指令学习

一 、AT指令 测试指令&#xff1a;ATCWMODE? 参数及取值范围 cwmode&#xff08;1-3&#xff09; 查询指令&#xff1a; ATCWMODE&#xff1f; 当前cwmode的取值 3 设置指令&#xff1a; ATCWMODE3 设置当前的cwmode为 3 1、station 模式 连接到其他wifi 2、softA…

Unity实战案例全解析:RTS游戏的框选和阵型功能(5)阵型功能 优化

前篇&#xff1a;Unity实战案例全解析&#xff1a;RTS游戏的框选和阵型功能&#xff08;4&#xff09;阵型功能-CSDN博客 本案例来源于unity唐老狮&#xff0c;有兴趣的小伙伴可以去泰克在线观看该课程 我只是对重要功能进行分析和做出笔记分享&#xff0c;并未无师自通&#x…

SpringBoot3+Druid YAML配置

背景 Druid连接池是阿里巴巴开源的数据库连接池项目。Druid连接池为监控而生&#xff0c;内置强大的监控功能&#xff0c;监控特性不影响性能。功能强大&#xff0c;能防SQL注入&#xff0c;内置Loging能诊断Hack应用行为。现在已经SpringBoot3&#xff0c;Druid的配置也需要随…

Yolov11项目实战1:道路缺陷检测系统设计【Python源码+数据集+运行演示】

一、项目背景 随着城市化进程的加速和交通网络的不断扩展&#xff0c;道路维护成为城市管理中的一个重要环节。道路缺陷&#xff08;如裂缝、坑洞、路面破损等&#xff09;不仅影响行车安全&#xff0c;还会增加车辆的磨损和维修成本。传统的道路缺陷检测方法主要依赖人工巡检…

HarmonyOS/OpenHarmony Audio 实现音频录制及播放功能

关键词&#xff1a;audio、音频录制、音频播放、权限申请、文件管理 在app的开发过程中时常会遇见一些需要播放一段音频或进行语音录制的场景&#xff0c;那么本期将介绍如何利用鸿蒙 audio 模块实现音频写入和播放的功能。本次依赖的是 ohos.multimedia.audio 音频管理模块&am…

前缀和算法详解

对于查询区间和的问题&#xff0c;可以预处理出来一个前缀和数组 dp&#xff0c;数组中存储的是从下标 0 的位置到当前位置的区间和&#xff0c;这样只需要通过前缀和数组就可以快速的求出指定区间的和了&#xff0c;例如求 l ~ r 区间的和&#xff0c;就可以之间使用 dp[l - 1…

河南做网站与SEO:如何提升搜索引擎排名

河南做网站与SEO&#xff1a;如何提升搜索引擎排名 在当今数字化时代&#xff0c;越来越多的企业意识到互联网的重要性&#xff0c;特别是在河南这样一个快速发展的地区&#xff0c;建立一个优秀的网站已经成为企业发展的必要条件。而在建立网站的同时&#xff0c;SEO&#xff…

Spring Gateway学习

系列文章目录 JavaSE基础知识、数据类型学习万年历项目代码逻辑训练习题代码逻辑训练习题方法、数组学习图书管理系统项目面向对象编程&#xff1a;封装、继承、多态学习封装继承多态习题常用类、包装类、异常处理机制学习集合学习IO流、多线程学习仓库管理系统JavaSE项目员工…

高性能防静电主轴4033 AC-ESD 在线路板切割中的非凡表现

随着电子产品的日益小型化/集成化&#xff0c;线路板的制造也面临着更高的挑战。线路板分板作为电子制造流程中的关键环节&#xff0c;其效率和精度直接影响到最终产品的质量和市场竞争力。因此专用的高性能防静电主轴SycoTec 4033 AC-ESD凭借其卓越的性能&#xff0c;成为众多…