0

0

C++怎么实现拓扑排序_C++有向无环图应用【实战】

冰火之心

冰火之心

发布时间:2026-03-16 18:37:03

|

935人浏览过

|

来源于php中文网

原创

Kahn算法用std::queue和入度数组实现拓扑排序:建邻接表vector、初始化indeg数组,入度0节点入队;每次出队时遍历邻接点并减其入度,为0则入队;若队空但未输出全部节点,则有环。

c++怎么实现拓扑排序_c++有向无环图应用【实战】

怎么用 std::queue 和入度数组做拓扑排序

拓扑排序本质是按依赖关系线性化有向无环图(DAG),C++里最稳的实现就是「Kahn 算法」:统计每个节点入度,把入度为 0 的扔进 std::queue,逐个取出并减少邻居入度。

关键不是“能不能写”,而是别漏掉三个动作:

  • 建图时用 vector<vector>></vector> 存邻接表,别用 mapset —— 没必要,还慢
  • 入度数组大小必须和节点编号范围对齐,比如节点是 0~n-1,就开 vector<int> indeg(n)</int>;要是节点编号稀疏(比如 1、5、100),得先离散化或用 unordered_map 管理入度
  • 每次从队列取节点后,要遍历它的所有邻接点,并对每个邻接点执行 indeg[v]--;只有当 indeg[v] == 0 时才入队 —— 少判这一句,结果就漏节点

遇到 std::queue 为空但还有节点没输出,说明图有环

这是 Kahn 算法自带的环检测能力。只要最终输出的节点数

常见误判场景:

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

  • 输入边时重复加了同一条边,导致某节点入度被多减,提前变成负数,后续判断失效
  • 图不连通,但你只从 0 号节点开始 BFS —— 不行,必须初始化时就把所有入度为 0 的节点都塞进队列
  • 节点编号从 1 开始,但入度数组只开了 n 大小却用 indeg[i] 访问,i=1 时越界未报错,但值是脏数据

调试时直接打印 result.size()n 对比,比看中间过程更可靠。

std::stack 替换 std::queue 会改变结果顺序,但仍是合法拓扑序

Kahn 算法本身不规定「选哪个入度为 0 的节点先处理」,所以用栈只是改变了贪心策略:从「先进先出」变成「后进先出」,结果仍是拓扑序,只是不同。

知我AI
知我AI

一款多端AI知识助理,通过一键生成播客/视频/文档/网页文章摘要、思维导图,提高个人知识获取效率;自动存储知识,通过与知识库聊天,提高知识利用效率。

下载

影响点很实际:

  • 如果题目要求字典序最小的拓扑序(比如节点是字符或带权重),就不能用普通队列或栈,得换成 priority_queue(小根堆)
  • stack 后,同一轮多个可选节点中,编号大的会先被处理,容易让人误以为“顺序错了”——其实没错,只是不唯一
  • 性能上没差别,stackqueue 都是 O(1) 均摊操作

邻接表用 vector<vector>></vector> 还是 vector<unordered_set>></unordered_set>

除非你需要频繁查「u→v 是否有边」,否则一律用 vector<vector>></vector>。理由很实在:

  • 拓扑排序过程中,你只遍历出边,不查边是否存在;用 unordered_set 白占内存、白耗哈希时间
  • 如果输入含重边(比如连续两次 addEdge(1,2)),用 vector 会存两条,导致 indeg[2] 被多减一次 —— 所以读边时得自己去重,或用 set 辅助判重
  • 极端情况:节点数 10⁵,边数 10⁵,unordered_set 每个节点额外开哈希桶,内存可能翻倍,且 cache 不友好

真正需要查边存在的场景极少,比如动态删边再拓扑——那已经不是标准 Kahn 能解决的问题了。

拓扑排序难的从来不是代码几行,而是建图前想清楚节点怎么编号、重边怎么处理、空图或单点怎么兜底。这些地方一松劲,跑样例全过,交上去 WA 在第 37 个测试点。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
堆和栈的区别
堆和栈的区别

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

448

2023.07.18

堆和栈区别
堆和栈区别

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

606

2023.08.10

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

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

448

2023.07.18

堆和栈区别
堆和栈区别

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

606

2023.08.10

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

77

2025.09.05

golang map相关教程
golang map相关教程

本专题整合了golang map相关教程,阅读专题下面的文章了解更多详细内容。

41

2025.11.16

golang map原理
golang map原理

本专题整合了golang map相关内容,阅读专题下面的文章了解更多详细内容。

67

2025.11.17

java判断map相关教程
java判断map相关教程

本专题整合了java判断map相关教程,阅读专题下面的文章了解更多详细内容。

47

2025.11.27

chatgpt使用指南
chatgpt使用指南

本专题整合了chatgpt使用教程、新手使用说明等等相关内容,阅读专题下面的文章了解更多详细内容。

0

2026.03.16

热门下载

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

精品课程

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

共94课时 | 11.5万人学习

C 教程
C 教程

共75课时 | 5.5万人学习

C++教程
C++教程

共115课时 | 22.2万人学习

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

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