深度优先搜索 (DFS) 详解

1. 什么是深度优先搜索?

深度优先搜索(Depth-First Search, DFS)是一种用于遍历或搜索树或图的算法。这个算法会尽可能深地搜索树的分支。当节点v的所在边都已被探寻过,搜索将回溯到发现节点v的那条边的起始节点。这一过程一直进行到已发现从源节点可达的所有节点为止。如果还存在未被发现的节点,则选择其中一个作为源节点并重复以上过程,整个进程反复进行直到所有节点都被访问为止。

2. 算法思想

DFS 的核心思想是 “一路走到黑,不撞南墙不回头”。想象你在一个迷宫里,你选择一条路一直走,直到遇到死胡同,然后返回到上一个路口,选择另一条你没走过的路继续探索。

为了实现这一点,我们需要一个机制来记录我们访问过的节点,以避免重复访问和无限循环。通常我们使用一个 visited 集合来存储已访问的节点。

DFS 主要有两种实现方式:

  1. 递归 (Recursion):利用函数调用栈来隐式地管理节点的访问顺序。这是最直观和常见的实现方式。
  2. 迭代 (Iteration):显式地使用一个栈 (Stack) 数据结构来模拟递归的过程。

3. Python 代码示例

我们先定义一个图。这里使用邻接表(Adjacency List)来表示图,它是一个字典,其中键是节点,值是与该节点相邻的节点列表。

# 定义一个图的邻接表表示
graph = {'A': ['B', 'C'],'B': ['A', 'D', 'E'],'C': ['A', 'F'],'D': ['B'],'E': ['B', 'F'],'F': ['C', 'E']
}print("图的邻接表表示:")
for node, neighbors in graph.items():print(f"{node}: {neighbors}")
图的邻接表表示:
A: ['B', 'C']
B: ['A', 'D', 'E']
C: ['A', 'F']
D: ['B']
E: ['B', 'F']
F: ['C', 'E']

3.1 递归实现

visited_recursive = set() # 用于存储已访问过的节点def dfs_recursive(graph, node):"""递归实现的深度优先搜索"""if node not in visited_recursive:print(node, end=' ') # 访问节点visited_recursive.add(node) # 标记为已访问# 递归访问邻居节点for neighbor in graph[node]:dfs_recursive(graph, neighbor)print("递归DFS遍历结果: ")
dfs_recursive(graph, 'A') # 从节点 'A' 开始遍历
递归DFS遍历结果: 
A B D E F C 

3.2 迭代实现

def dfs_iterative(graph, start_node):"""迭代实现的深度优先搜索"""visited_iterative = set() # 存储已访问的节点stack = [start_node]      # 用列表模拟栈,初始放入起始节点while stack: # 当栈不为空时循环node = stack.pop() # 弹出栈顶元素if node not in visited_iterative:print(node, end=' ') # 访问节点visited_iterative.add(node) # 标记为已访问# 将邻居节点逆序压入栈中,以保证访问顺序与递归版本一致# (因为pop会弹出最后的元素,所以逆序放入后,第一个邻居会最后被pop)for neighbor in reversed(graph[node]):if neighbor not in visited_iterative:stack.append(neighbor)print("迭代DFS遍历结果: ")
dfs_iterative(graph, 'A') # 从节点 'A' 开始遍历
迭代DFS遍历结果: 
A B D E F C 

4. 复杂度分析

假设图有 V 个节点 (Vertices) 和 E 条边 (Edges)。

  • 时间复杂度: O(V + E)。

    • 每个节点最多被访问一次 (入栈/递归调用一次),所以是 O(V)。
    • 每条边最多被探索一次,所以是 O(E)。
    • 总的时间复杂度是 O(V + E)。
  • 空间复杂度: O(H),其中 H 是图中的最大深度。

    • 在递归实现中,空间复杂度取决于递归调用的最大深度。在最坏的情况下(例如一个链状的图),H 可能等于 V。
    • 在迭代实现中,空间复杂度取决于栈中存储的节点数,同样在最坏情况下是 O(V)。

5. 应用场景

DFS 是一个非常基础且强大的算法,应用广泛,例如:

  1. 寻找路径:查找从一个节点到另一个节点是否存在路径。
  2. 检测环:判断图中是否存在环。在有向图和无向图中都可以使用。
  3. 拓扑排序 (Topological Sorting):对于有向无环图 (DAG),可以得到一个线性的节点排序。
  4. 寻找连通分量 (Connected Components):在无向图中,可以用DFS找到所有的连通子图。
  5. 解决迷宫问题:从起点开始,使用DFS探索所有可能的路径直到找到出口。

代码链接:
https://github.com/zhouruiliangxian/Awesome-demo/tree/main/Data_Structures_and_Algorithms

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

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

相关文章

文心4.5开源大模型的使用和部署

前言 就在今天,文心4.5模型开源了,不是一个,而是整个系列模型正式开源。很突然,我都震惊了。文心4.5系列开源模型共10款,涵盖了激活参数规模分别为47B 和3B 的混合专家(MoE)模型(最…

HarmonyOs开发之——TypeScript介绍、入门,及 TypeScript、JavaScript、ArkTs的具体区别解读。

HarmonyOs开发之——TypeScript介绍、入门,及 TypeScript、JavaScript、ArkTs的具体区别解读。 一、 开发语言介绍: TypeScript是JavaScript的超集,ArkTS则是TypeScript的超集。ArkTs是 HarmonyOs的主力开发语言,它在TypeScript…

《JMS事务性会话彻底解析:消息监听中的 commit、rollback 和幂等设计》

大家好,我是G探险者! 📌 场景引入 在实际项目中,我们常常面临以下挑战: 监听 MQ 消息失败了,希望自动重试?消费 MQ 消息后,要写数据库,但中间报错了?消息处…

vue3 el-table 列增加 自定义排序逻辑

在 Vue 3 中使用 Element Plus 的 <el-table> 组件时&#xff0c;如果你想增加自定义排序逻辑&#xff0c;可以通过以下几个步骤实现&#xff1a; 1. 使用 default-sort 属性 首先&#xff0c;你可以在 <el-table> 组件上使用 default-sort 属性来指定默认的排序…

ISP Pipeline(7): Gamma Correction 伽马校正

AI_Plays/ISP/Fast_ISP_Progress.ipynb at main ameengee/AI_Plays GitHub Gamma Correction&#xff08;伽马校正&#xff09;是图像处理中的一个重要步骤&#xff0c;目的是调整图像的亮度&#xff0c;使其更符合人眼的感知或显示设备的特性。 为什么需要 Gamma Correcti…

AI提取伴奏,实现卡拉OK效果 —— 「suno api/luno api/kuka api」

导读 喜欢唱歌&#xff0c;却总苦于找不到纯净的伴奏&#xff1f;或者你想把喜欢的歌曲翻唱一遍&#xff0c;却被人声干扰搞得头大&#xff1f;现在&#xff0c;AI技术已经悄悄解决了这个问题。借助AI智能工具&#xff0c;你可以轻松提取任何一首歌的伴奏&#xff0c;享受宛如…

pip介绍

pip是什么&#xff1f; pip&#xff08;Pip Installs Packages&#xff09;是Python的官方管理工具&#xff0c;用于安装、升级、卸载和管理Python第三方库及其依赖关系。它是Python生态系统的核心组件&#xff0c;通过连接PyPI&#xff08;Python Package Index&#xff09;这…

机器学习20-线性网络思考

机器学习20-线性网络思考 针对线性网络的基础问题&#xff0c;使用基础示例进行解释 1-核心知识点 1-线性模型家族的线性回归和逻辑回归分别是什么&#xff0c;线性模型家族还有没有其他的模型 线性模型家族是一系列基于线性假设的统计模型&#xff0c;它们假设因变量和自变量…

【科研绘图系列】R语言绘制世界地图分布(world map)

禁止商业或二改转载,仅供自学使用,侵权必究,如需截取部分内容请后台联系作者! 文章目录 介绍加载R包数据下载导入数据数据预处理准备画图画图总结系统信息介绍 本教程旨在通过R语言及其相关地理空间分析包,展示如何对环境数据进行空间聚类分析,并将结果可视化。教程从读…

Armbian 25.5.1 Noble Gnome 开启远程桌面功能

sudo apt install gnome-remote-desktop ----长话短说 故障表现 Ubuntu 25版本点击远程桌面功能没有任何反应, WIN_20250630_00_53_24_Pro 最后 armbian 官方社区充满了傲慢,一言不合就关闭话题,问题都没有解决就给我关闭了 最后检索到英文网站,说到了这么一句话,检查远程桌…

嵌入式 Linux 入门:从裸机到系统级开发的第一步

随着嵌入式系统应用的不断深入&#xff0c;很多 MCU 项目开发者会在某个阶段遇到瓶颈&#xff1a;系统越来越复杂、任务越来越多、通信越来越频繁、性能要求越来越高。 这时候&#xff0c;从 MCU / RTOS 过渡到 嵌入式 Linux 开发 就成为一次技术升级的关键转折点。 本文将带…

详解 Blazor 组件传值

父子组件传值 在 Blazor 中&#xff0c;组件之间的通信可以通过 [Parameter] 参数和 EventCallback<T> 事件回调实现。下面分别给出 父组件传递值给子组件 和 子组件传递值给父组件 的简单示例。 1.1 父组件传递值给子组件 步骤&#xff1a; 在子组件中定义 public 属…

力扣热题100再刷

160.相交链表 读一遍A&#xff0c;一个set存节点&#xff0c;遍历B的时候判断即可。复习下set的STL&#xff1a;set有set和unordered_set&#xff0c;同样有insert&#xff0c;find&#xff0c;count&#xff0c;对于set而言&#xff0c;自动从小到大排序&#xff0c;还有&…

MySQL常用函数性能优化及索引影响分析

MySQL 常用函数性能优化指南&#xff08;含索引影响分析&#xff09; 以下是 MySQL 函数使用指南&#xff0c;新增性能影响评级、索引失效分析和优化方案&#xff0c;帮助您高效使用函数&#xff1a; &#x1f4dc; 一、字符串处理函数&#xff08;含性能分析&#xff09; 函…

莫队(基础版)优雅的暴力

莫队算法是一种离线算法&#xff0c;常用于高效处理区间查询问题。它通过合理排序和移动左右端点来减少时间复杂度。 基本思想 莫队算法的核心思想是将所有查询离线排序&#xff01;&#xff01;&#xff08;找出一个过起来最快的查询顺序&#xff09;&#xff0c;然后通过移动…

✨ Python 高级定制 | 美化 Word 表格边框与样式(收货记录增强版)

之前我们完成了 Excel 数据提取、Word 表格写入与合并&#xff0c;现在继续 为 Word 表格添加高级样式 装扮&#xff0c;包括单元格边框、背景填色、居中对齐、粗体、高亮行/列等&#xff0c;进一步增强表格的可读性与专业性。 &#x1f58c;️ 样式设置函数 1. 设置单元格边框…

Clickhouse源码分析-TTL执行流程

第一种情况&#xff1a;无ttl_only_drop_parts配置 总体示例以及说明 如果没有ttl_only_drop_parts的配置&#xff0c;过期数据的删除&#xff08;这里是删除&#xff0c;是将过期的数据从这个part删除&#xff0c;并将过期的数据构成一个part&#xff0c;这个过期的part标记…

elementui修改radio字体的颜色和圆圈的样式

改完 <div class"choose"><el-radio-group v-model"radioNum"><el-radio label"1" size"large">Option 1</el-radio><el-radio label"2" size"large">Option 2</el-radio>&l…

力扣3381. 长度可被 K 整除的子数组的最大元素和

由于数据范围是2*10^5所以必然是遍历一次&#xff0c;子数组必定要用到前缀和&#xff0c;之前的题目中总是遇到的是子数组的和能不能被k整除&#xff0c;而这里不一样的是子数组的长度能不能被k整除&#xff0c;如果单纯的枚举长度必定超时&#xff0c;而看看题解得出的思路&a…

基于SSM的勤工助学系统的设计与实现

第1章 摘要 基于SSM框架的勤工助学系统旨在为学生、用工部门和管理员提供高效便捷的管理平台。系统包括学生端、用工部门端和管理员端&#xff0c;涵盖了从岗位发布、申请审核、工时记录、薪资管理到数据统计等完整的功能需求。 学生可以通过系统首页浏览最新的岗位信息和公告&…