0

0

C# 深度优先搜索方法 C#如何用递归和栈实现DFS

畫卷琴夢

畫卷琴夢

发布时间:2026-03-15 10:17:03

|

797人浏览过

|

来源于php中文网

原创

递归DFS需在进入前标记visited以防重复访问和栈溢出,迭代DFS须逆序压栈并出栈时标记;需路径时递归更易回溯,无向图需防父节点回溯。

c# 深度优先搜索方法 c#如何用递归和栈实现dfs

递归实现 DFS 时节点访问顺序容易出错

递归写法最直观,但新手常把 visited 标记位置放错,导致重复访问或漏访问。必须在进入递归前就标记当前节点,而不是在循环里或递归调用后才标。

  • 正确顺序:标记 → 访问 → 遍历邻接点 → 递归
  • 错误写法示例:foreach (var next in neighbors) { DFS(next); visited[next] = true; } —— 这会导致子节点被多次递归进入
  • 图中存在环时,不提前标记 visited 会直接栈溢出
  • 递归深度受限于 .NET 默认线程栈(约 1MB),超深树(如 >10000 层)可能抛 StackOverflowException(.NET 6+ 在非 Windows 上不可捕获)

用 Stack 手动模拟递归要反向压栈

迭代版 DFS 用 Stack<T> 实现,关键在于:邻接节点必须「逆序」入栈,才能保证和递归版一致的访问顺序(比如左子树先于右子树)。

  • 若按自然顺序压栈(如先压左再压右),弹出时变成右先访问 —— 和递归行为不一致
  • 正确做法:for (int i = neighbors.Count - 1; i >= 0; i--) stack.Push(neighbors[i]);
  • visited 仍需在出栈后、访问前立即标记,避免同一节点被多次压入
  • 注意不要在入栈时就标记,否则未访问的节点被提前“锁定”,会漏掉路径

DFS 返回路径时,递归比栈更容易回溯

当需要返回从起点到目标的完整路径(如迷宫求解),递归天然支持回溯:函数返回即自动“撤出”当前节点;而栈版本需额外维护父指针或路径栈,逻辑更重。

小微助手
小微助手

微信推出的一款专注于提升桌面效率的助手型AI工具

下载
  • 递归可直接用局部 List<T> 累积路径,命中目标后直接返回该列表
  • 栈实现需为每个入栈节点保存其前驱(如用 Dictionary<T, T>),或让栈元素是元组 (node, pathSoFar),内存开销明显上升
  • 若只判断连通性(bool 返回值),两者性能差异不大;但涉及路径重建,递归代码更紧凑、不易错

遍历无向图时记得跳过父节点防自环

对树或无向图做 DFS,若不加限制,从子节点回看父节点会被当作新邻居,造成“原地振荡”。这不是 bug,而是图结构特性导致的,必须显式规避。

  • 递归版可在参数中传入 parent,遍历时跳过:if (next != parent) DFS(next, current);
  • 栈版本则需在压栈前检查:if (next != currentNode) stack.Push(next);(仅适用于已知父子关系的场景)
  • 更通用的做法仍是依赖 visited 数组——只要节点已访问过就跳过,但前提是图不含重边且初始化充分
  • 这个细节在邻接表实现中极易被忽略,一跑测试用例就无限循环

实际用哪一种,取决于你是否担心栈深度、是否需路径信息、以及图是否有向。递归写起来快,但生产环境面对深层图建议切栈实现;而一旦要返回路径,又得权衡可读性和内存占用——这些不是理论选择,是上线前真会卡住你的点。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

847

2023.08.22

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

203

2023.11.20

php中foreach用法
php中foreach用法

本专题整合了php中foreach用法的相关介绍,阅读专题下面的文章了解更多详细教程。

268

2025.12.04

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1051

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

615

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

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

335

2025.08.29

C++中int的含义
C++中int的含义

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

235

2025.08.29

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

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

448

2023.07.18

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

49

2026.03.13

热门下载

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

精品课程

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

共94课时 | 11.4万人学习

C 教程
C 教程

共75课时 | 5.5万人学习

C++教程
C++教程

共115课时 | 22万人学习

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

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