0

0

Python图的DFS怎么写_深度优先搜索与栈/递归遍历

P粉602998670

P粉602998670

发布时间:2026-03-11 18:21:11

|

470人浏览过

|

来源于php中文网

原创

递归dfs直观但易栈溢出,适合小图;迭代dfs用栈更可控,需逆序压栈保证顺序;邻接表优于邻接矩阵;dfs不自动遍历全图,需外层循环处理多连通分量。

python图的dfs怎么写_深度优先搜索与栈/递归遍历

DFS递归写法最直观,但要注意栈溢出风险

Python默认递归深度只有1000层,图节点多或深度大时,RecursionError: maximum recursion depth exceeded几乎必现。这不是代码写错了,是语言限制。

递归DFS适合小图、教学演示或已知结构浅的场景(比如二叉树)。核心就是「访问当前节点 → 遍历未访问邻居 → 递归处理每个邻居」。

  • visited集合记录已访问节点,避免环导致死循环
  • 邻接表用dictlist都行,但查邻居必须O(1),别在循环里反复.index()
  • 如果需要路径回溯(比如找两点间路径),递归参数要带path列表,但注意别直接传path.append(x)——那会改原列表,得传path + [x]
def dfs_recursive(graph, start, visited=None):
    if visited is None:
        visited = set()
    visited.add(start)
    print(start)  # 或存入结果
    for neighbor in graph.get(start, []):
        if neighbor not in visited:
            dfs_recursive(graph, neighbor, visited)
    return visited

用栈模拟DFS更可控,适合真实项目

显式用list当栈(append/pop())能完全避开递归限制,逻辑也更贴近算法本质:后进先出,优先深入最新发现的分支。

关键不是“能不能写”,而是「顺序是否符合预期」——Python的list.pop()弹尾,所以压栈顺序得反着来,否则遍历顺序和递归版不一致(比如想按字母序访问邻居,就得先压'z'再压'a')。

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

银河易创
银河易创

一站式AIGC创作平台,集成GPT-3.5、GPT-4、文心一言等对话模型、Midjourney、DallE等绘画工具、AI音乐、AI视频和AI PPT等功能!

下载
  • 栈里存的是待处理节点,不是边或状态;每次pop()一个,立刻标记visited
  • 不要在for循环里动态修改正在遍历的graph[start],容易漏节点
  • 如果图用defaultdict(list),记得处理graph[node]可能为空的情况,避免KeyError
def dfs_iterative(graph, start):
    stack, visited = [start], set()
    while stack:
        node = stack.pop()
        if node not in visited:
            visited.add(node)
            print(node)
            # 注意:逆序压栈才能保持和递归相同的访问顺序
            for neighbor in reversed(graph.get(node, [])):
                if neighbor not in visited:
                    stack.append(neighbor)
    return visited

邻接矩阵下DFS性能差,别硬套

list of list存图时,查一个节点的所有邻居要O(V)扫整行,而邻接表是O(degree)。V=1000时,单次遍历邻居就多999次无效判断。

除非图非常稠密(边数接近)且内存极宽裕,否则别用矩阵存图再跑DFS。更糟的是,很多人写成for j in range(len(matrix)):然后if matrix[i][j]: ...,这比用enumerate还慢。

  • 真要用矩阵,至少预处理每行的非零索引:neighbors[i] = [j for j, v in enumerate(matrix[i]) if v],只做一次
  • 但更推荐直接转邻接表:graph = {i: [] for i in range(n)}; for i in range(n): for j in range(n): if matrix[i][j]: graph[i].append(j)
  • 稀疏图下,邻接表内存占用也小得多——10k节点、100条边,矩阵要100MB,邻接表不到1MB

遇到环、孤立点、不连通图时的典型表现

DFS本身不保证访问全部节点,它只从起点出发。常见错误是调用一次dfs(graph, 0)就以为全图遍历完了,结果孤立点或另一连通分量里的节点根本没进visited

真正健壮的实现得外层套个循环:for node in all_nodes:,跳过已访问的。但这不是DFS的问题,是使用者没理解「DFS是遍历策略,不是图算法全貌」。

  • visited必须是全局集合,不能每次递归都新建;栈版本同理,visited要在while外初始化
  • 无向图边存两份(a→bb→a),但visited检查到位的话,不会重复递归——这点常被怀疑“为什么没死循环”,其实是机制在起作用
  • 有向图中,从A出发搜不到B,不代表B不可达,只是方向反了;这时候需要考虑reverse_graph或换起点

实际写的时候,递归看着清爽,但生产环境一律用栈;图结构不确定时,先打印len(graph)max(len(v) for v in graph.values())看规模,再决定要不要加sys.setrecursionlimit()——加了也可能OOM,不如一开始就选对路。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

846

2023.08.22

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

106

2023.09.25

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

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

443

2023.07.18

堆和栈区别
堆和栈区别

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

605

2023.08.10

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

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

443

2023.07.18

堆和栈区别
堆和栈区别

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

605

2023.08.10

append用法
append用法

append是一个常用的命令行工具,用于将一个文件的内容追加到另一个文件的末尾。想了解更多append用法相关内容,可以阅读本专题下面的文章。

348

2023.10.25

python中append的用法
python中append的用法

在Python中,append()是列表对象的一个方法,用于向列表末尾添加一个元素。想了解更多append的更多内容,可以阅读本专题下面的文章。

1080

2023.11.14

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

3

2026.03.11

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.5万人学习

Django 教程
Django 教程

共28课时 | 4.9万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.9万人学习

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

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