0

0

C++怎么实现最小生成树_C++Prim Kruskal教程【图论】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-02-24 14:31:02

|

180人浏览过

|

来源于php中文网

原创

prim算法用priority_queue实现时易出错,因c++优先队列不支持降键操作,需在出队时用inmst[]检查是否已加入mst,避免重复处理;队列应存{weight, node}并用greater实现小根堆;起点入队{0,start},其余点由松弛过程加入。

c++怎么实现最小生成树_c++prim kruskal教程【图论】

Prim 算法用 priority_queue 实现时,为什么老是多加边或漏加点?

Prim 的核心是维护「已连通集合」和「待选边」,但 C++ 的 priority_queue 默认不支持修改堆内元素优先级,所以不能像伪代码里那样“更新邻接点权重”。一旦某个点被加入 MST 后又被更小权边再次推入队列,就会重复处理。

  • 必须在出队时检查该点是否已在 MST 中——用布尔数组 inMST[] 标记,出队后立刻判断,不是一进队就标记
  • priority_queue 要存 {weight, node},且按 weight 升序;C++ 默认大根堆,得用 greater 或自定义比较
  • 图要用邻接表(vector<vector int>>></vector>),别用邻接矩阵,稀疏图下空间和时间都吃不消
  • 起点初始化:把起点的 {0, start} 推入,其他点先不入队,靠松弛过程自然加入

Kruskal 需要排序边,sort() 传什么比较函数才不会崩?

边结构体如果只重载 operator,容易因成员未初始化或比较逻辑错位导致 <code>sort 行为未定义(尤其在含负权边或自环时)。最稳的方式是用 lambda 显式比权重。

  • 边建议定义为 struct Edge { int u, v, w; };,别用 tuple,可读性和调试成本低得多
  • sort(edges.begin(), edges.end(), [](const Edge& a, const Edge& b) { return a.w —— 必须用 <code>a.w ,不是 <code>,否则可能违反 strict weak ordering
  • 排序前确保 edges 不含自环(u == v)和重复边,Kruskal 对这些没做防御
  • 如果图不连通,Kruskal 结束后 MST 边数会小于 n-1,得手动检查,不能默认成功

并查集写 unionSet() 时路径压缩和按秩合并哪个更重要?

两者都重要,但路径压缩(find() 里改父节点)是必须的;按秩合并(unionSet() 里比 rank[])能进一步压平树高,但漏掉它顶多让单次操作退化到 O(log n),而没路径压缩可能直接 O(n)。

Play.ht
Play.ht

根据文本生成多种逼真的语音

下载
  • find(x) 必须写成递归或带循环+赋值的形式:parent[x] = find(parent[x]);,不能只返回 find(parent[x])
  • rank[] 初始全 0,合并时只在两棵树 rank 相等时才给新根 rank+1;别用 size 合并(除非你真需要 size 信息)
  • 注意数组索引:节点编号从 0 还是 1 开始?parent 数组大小得是 n+1n,错一位就访问越界
  • 并查集对象别放在函数栈里传值复制,用引用或全局静态,否则每次调用都是新实例

两种算法选哪个?看图密度还是看数据范围?

看边数 m 和点数 n 的比值,不是看“稠密”“稀疏”的模糊描述。实际编码中,m 基本算稀疏,<code>m > n²/4 才值得怀疑是否真需要 Prim。

立即学习C++免费学习笔记(深入)”;

  • Kruskal 时间复杂度 O(m log m),适合 m ≤ 1e5;超过这量级,排序本身成瓶颈,且常数大
  • Prim + priority_queue 是 O(m log n),但隐式建堆开销明显;若用斐波那契堆理论更优,但 C++ 没原生实现,别折腾
  • 如果题目强制要求“从某点开始构造”,或者边权动态变化(虽少见),Prim 更自然;Kruskal 天然无起点概念
  • 内存敏感场景(嵌入式、OJ 内存限制严),Kruskal 只需存边和并查集,比 Prim 的邻接表 + 堆省空间

边界情况比教科书写的多:重边要留最小那条、自环直接扔、不连通图得提前判,这些地方一漏,输出就不是最小生成树,而是某棵生成树。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
sort排序函数用法
sort排序函数用法

sort排序函数的用法:1、对列表进行排序,默认情况下,sort函数按升序排序,因此最终输出的结果是按从小到大的顺序排列的;2、对元组进行排序,默认情况下,sort函数按元素的大小进行排序,因此最终输出的结果是按从小到大的顺序排列的;3、对字典进行排序,由于字典是无序的,因此排序后的结果仍然是原来的字典,使用一个lambda表达式作为key参数的值,用于指定排序的依据。

404

2023.09.04

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

365

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

200

2025.07.04

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

422

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

595

2023.08.10

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

470

2023.08.14

苹果官网入口与在线访问指南_中国站点快速直达与iPhone查看方法
苹果官网入口与在线访问指南_中国站点快速直达与iPhone查看方法

本专题汇总苹果官网最新可用入口及中国站点访问方式,涵盖官网直达链接、iPhone官方页面查看方法与常见访问说明,帮助用户快速进入苹果官方网站,便捷了解产品信息与官方服务。

4

2026.02.24

Asianfanfics官网入口与访问指南_AFF官方平台最新登录地址
Asianfanfics官网入口与访问指南_AFF官方平台最新登录地址

本专题系统整理Asianfanfics(AFF)官方网站最新可用入口,涵盖官方平台最新直达地址、官网登录方式及中文访问指引,帮助用户快速、安全地进入AFF平台浏览与使用相关内容。

7

2026.02.24

Go分布式服务治理与链路追踪实践
Go分布式服务治理与链路追踪实践

本专题围绕 Go 语言在分布式系统中的服务治理实践展开,系统讲解服务注册与发现、配置中心、限流熔断、灰度发布以及分布式链路追踪方案。通过完整项目示例,帮助开发者构建可观测性完善、稳定性高、易扩展的 Go 微服务架构体系,提升线上系统问题定位与运维效率。

1

2026.02.24

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 10万人学习

C 教程
C 教程

共75课时 | 4.9万人学习

C++教程
C++教程

共115课时 | 19.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号