当前位置: 首页 > article >正文

【数据结构与算法】第38篇:图论(二):深度优先搜索(DFS)与广度优先搜索(BFS)

一、图遍历的基本概念1.1 为什么需要遍历和树一样图也需要一种方式“访问”所有顶点。但图可能有环所以需要标记已访问的顶点避免重复访问。1.2 两种遍历方式遍历方式核心思想数据结构DFS一条路走到底回溯栈递归BFS一层层向外扩展队列二、深度优先搜索DFS2.1 算法思想从一个顶点出发访问它的一个邻接点再访问该邻接点的邻接点……直到无法继续然后回溯。示例图text0 — 1 | | 2 — 3从0出发的DFS0 → 1 → 3 → 2取决于邻接顺序2.2 递归实现c#include stdio.h #include stdlib.h #define MAX_VERTICES 100 // 邻接矩阵图 typedef struct { int matrix[MAX_VERTICES][MAX_VERTICES]; int vertexCount; } Graph; // 初始化图 void initGraph(Graph *g, int n) { g-vertexCount n; for (int i 0; i n; i) { for (int j 0; j n; j) { g-matrix[i][j] 0; } } } // 添加边无向图 void addEdge(Graph *g, int u, int v) { g-matrix[u][v] 1; g-matrix[v][u] 1; } // DFS递归实现 void dfsRecursive(Graph *g, int vertex, int visited[]) { visited[vertex] 1; printf(%d , vertex); for (int i 0; i g-vertexCount; i) { if (g-matrix[vertex][i] 1 !visited[i]) { dfsRecursive(g, i, visited); } } } // DFS入口 void dfs(Graph *g, int start) { int visited[MAX_VERTICES] {0}; printf(DFS遍历序列: ); dfsRecursive(g, start, visited); printf(\n); } int main() { Graph g; initGraph(g, 4); addEdge(g, 0, 1); addEdge(g, 0, 2); addEdge(g, 1, 3); addEdge(g, 2, 3); dfs(g, 0); return 0; }运行结果textDFS遍历序列: 0 1 3 22.3 非递归实现显式栈c#include stdio.h #include stdlib.h #define MAX_VERTICES 100 // 栈结构 typedef struct { int data[MAX_VERTICES]; int top; } Stack; void initStack(Stack *s) { s-top -1; } int isEmpty(Stack *s) { return s-top -1; } void push(Stack *s, int val) { s-data[s-top] val; } int pop(Stack *s) { return s-data[s-top--]; } // DFS非递归 void dfsIterative(Graph *g, int start) { int visited[MAX_VERTICES] {0}; Stack stack; initStack(stack); push(stack, start); visited[start] 1; printf(DFS(栈)遍历序列: ); while (!isEmpty(stack)) { int vertex pop(stack); printf(%d , vertex); for (int i g-vertexCount - 1; i 0; i--) { if (g-matrix[vertex][i] 1 !visited[i]) { visited[i] 1; push(stack, i); } } } printf(\n); }三、广度优先搜索BFS3.1 算法思想从一个顶点出发先访问它的所有邻接点再访问这些邻接点的邻接点一层层向外扩展。示例图同上从0出发的BFS0 → 1 → 2 → 33.2 队列实现c#include stdio.h #include stdlib.h #define MAX_VERTICES 100 // 队列结构 typedef struct { int data[MAX_VERTICES]; int front; int rear; } Queue; void initQueue(Queue *q) { q-front q-rear 0; } int isEmpty(Queue *q) { return q-front q-rear; } void enqueue(Queue *q, int val) { q-data[q-rear] val; } int dequeue(Queue *q) { return q-data[q-front]; } // BFS void bfs(Graph *g, int start) { int visited[MAX_VERTICES] {0}; Queue queue; initQueue(queue); visited[start] 1; enqueue(queue, start); printf(BFS遍历序列: ); while (!isEmpty(queue)) { int vertex dequeue(queue); printf(%d , vertex); for (int i 0; i g-vertexCount; i) { if (g-matrix[vertex][i] 1 !visited[i]) { visited[i] 1; enqueue(queue, i); } } } printf(\n); } int main() { Graph g; initGraph(g, 4); addEdge(g, 0, 1); addEdge(g, 0, 2); addEdge(g, 1, 3); addEdge(g, 2, 3); bfs(g, 0); return 0; }运行结果textBFS遍历序列: 0 1 2 3四、完整代码演示邻接表版本c#include stdio.h #include stdlib.h #define MAX_VERTICES 100 // 邻接表节点 typedef struct EdgeNode { int vertex; struct EdgeNode *next; } EdgeNode; // 顶点节点 typedef struct { EdgeNode *first; } VertexNode; // 图结构 typedef struct { VertexNode vertices[MAX_VERTICES]; int vertexCount; } GraphList; // 初始化 void initGraphList(GraphList *g, int n) { g-vertexCount n; for (int i 0; i n; i) { g-vertices[i].first NULL; } } // 添加边无向图 void addEdgeList(GraphList *g, int u, int v) { EdgeNode *e1 (EdgeNode*)malloc(sizeof(EdgeNode)); e1-vertex v; e1-next g-vertices[u].first; g-vertices[u].first e1; EdgeNode *e2 (EdgeNode*)malloc(sizeof(EdgeNode)); e2-vertex u; e2-next g-vertices[v].first; g-vertices[v].first e2; } // DFS递归邻接表 void dfsListRecursive(GraphList *g, int vertex, int visited[]) { visited[vertex] 1; printf(%d , vertex); EdgeNode *p g-vertices[vertex].first; while (p ! NULL) { if (!visited[p-vertex]) { dfsListRecursive(g, p-vertex, visited); } p p-next; } } void dfsList(GraphList *g, int start) { int visited[MAX_VERTICES] {0}; printf(DFS(邻接表): ); dfsListRecursive(g, start, visited); printf(\n); } // 队列 typedef struct { int data[MAX_VERTICES]; int front, rear; } Queue; void initQueue(Queue *q) { q-front q-rear 0; } int isEmpty(Queue *q) { return q-front q-rear; } void enqueue(Queue *q, int val) { q-data[q-rear] val; } int dequeue(Queue *q) { return q-data[q-front]; } // BFS邻接表 void bfsList(GraphList *g, int start) { int visited[MAX_VERTICES] {0}; Queue q; initQueue(q); visited[start] 1; enqueue(q, start); printf(BFS(邻接表): ); while (!isEmpty(q)) { int vertex dequeue(q); printf(%d , vertex); EdgeNode *p g-vertices[vertex].first; while (p ! NULL) { if (!visited[p-vertex]) { visited[p-vertex] 1; enqueue(q, p-vertex); } p p-next; } } printf(\n); } int main() { GraphList g; initGraphList(g, 4); addEdgeList(g, 0, 1); addEdgeList(g, 0, 2); addEdgeList(g, 1, 3); addEdgeList(g, 2, 3); dfsList(g, 0); bfsList(g, 0); return 0; }运行结果textDFS(邻接表): 0 2 3 1 BFS(邻接表): 0 2 1 3注意由于邻接表使用头插法边的顺序与添加顺序相反所以遍历序列与邻接矩阵版本可能不同。五、DFS与BFS的对比5.1 遍历序列对比以图0-1, 0-2, 1-3, 2-3为例遍历方式序列邻接矩阵序列邻接表头插DFS0,1,3,20,2,3,1BFS0,1,2,30,2,1,35.2 核心区别对比项DFSBFS数据结构栈递归队列空间复杂度O(树高)O(最大宽度)最短路径不保证保证无权图连通分量可找出可找出环检测可检测可检测拓扑排序适用适用5.3 时间复杂度存储方式DFSBFS邻接矩阵O(V²)O(V²)邻接表O(VE)O(VE)六、应用场景应用推荐原因迷宫寻路最短路径BFS第一次到达即为最短拓扑排序DFS后序输出检测环DFS递归栈可追踪路径连通分量DFS/BFS均可二分图检测BFS按层染色强连通分量DFSTarjan/Kosaraju算法深度优先的搜索如八皇后DFS需要回溯七、完整代码带图构建和双遍历c#include stdio.h #include stdlib.h #define MAX_VERTICES 100 // 图结构邻接表 typedef struct EdgeNode { int vertex; struct EdgeNode *next; } EdgeNode; typedef struct { EdgeNode *first; } VertexNode; typedef struct { VertexNode vertices[MAX_VERTICES]; int vertexCount; } Graph; // 初始化 void initGraph(Graph *g, int n) { g-vertexCount n; for (int i 0; i n; i) { g-vertices[i].first NULL; } } // 添加边 void addEdge(Graph *g, int u, int v) { EdgeNode *e (EdgeNode*)malloc(sizeof(EdgeNode)); e-vertex v; e-next g-vertices[u].first; g-vertices[u].first e; e (EdgeNode*)malloc(sizeof(EdgeNode)); e-vertex u; e-next g-vertices[v].first; g-vertices[v].first e; } // DFS void dfs(Graph *g, int v, int visited[]) { visited[v] 1; printf(%d , v); EdgeNode *p g-vertices[v].first; while (p) { if (!visited[p-vertex]) { dfs(g, p-vertex, visited); } p p-next; } } void startDFS(Graph *g, int start) { int visited[MAX_VERTICES] {0}; printf(DFS: ); dfs(g, start, visited); printf(\n); } // BFS void bfs(Graph *g, int start) { int visited[MAX_VERTICES] {0}; int queue[MAX_VERTICES]; int front 0, rear 0; visited[start] 1; queue[rear] start; printf(BFS: ); while (front rear) { int v queue[front]; printf(%d , v); EdgeNode *p g-vertices[v].first; while (p) { if (!visited[p-vertex]) { visited[p-vertex] 1; queue[rear] p-vertex; } p p-next; } } printf(\n); } int main() { Graph g; initGraph(g, 6); // 构建图 addEdge(g, 0, 1); addEdge(g, 0, 2); addEdge(g, 1, 3); addEdge(g, 2, 4); addEdge(g, 3, 5); addEdge(g, 4, 5); startDFS(g, 0); bfs(g, 0); return 0; }运行结果textDFS: 0 2 4 5 3 1 BFS: 0 2 1 4 3 5八、小结这一篇我们学习了图的两种遍历方式遍历核心数据结构应用DFS深入到底回溯栈递归拓扑排序、环检测、连通分量BFS层层扩散按层队列最短路径无权图、二分图关键代码模板c// DFS模板 void dfs(Graph *g, int v, int visited[]) { visited[v] 1; for (邻接点 w) { if (!visited[w]) dfs(g, w, visited); } } // BFS模板 void bfs(Graph *g, int start) { 队列 q; visited[start] 1; q.push(start); while (q不空) { v q.pop(); for (邻接点 w) { if (!visited[w]) { visited[w] 1; q.push(w); } } } }下一篇我们讲最小生成树Prim算法与Kruskal算法。九、思考题为什么DFS通常用递归实现而BFS用队列实现可以互换吗在邻接矩阵表示的图中DFS和BFS的时间复杂度都是O(V²)哪个常数因子更小如果图是非连通的如何遍历所有顶点如何用DFS检测图中是否有环欢迎在评论区讨论你的答案。

相关文章:

【数据结构与算法】第38篇:图论(二):深度优先搜索(DFS)与广度优先搜索(BFS)

一、图遍历的基本概念1.1 为什么需要遍历和树一样,图也需要一种方式“访问”所有顶点。但图可能有环,所以需要标记已访问的顶点,避免重复访问。1.2 两种遍历方式遍历方式核心思想数据结构DFS一条路走到底,回溯栈(递归&…...

Chandra OCR完整教程:从单图测试到企业级应用,全流程实战解析

Chandra OCR完整教程:从单图测试到企业级应用,全流程实战解析 1. Chandra OCR核心能力解析 Chandra OCR是Datalab.to在2025年开源的一款革命性文档识别工具,与传统OCR相比具有三大突破性优势: 布局感知:不仅能识别文…...

5分钟快速上手:抖音无水印批量下载工具完整指南

5分钟快速上手:抖音无水印批量下载工具完整指南 【免费下载链接】douyin-downloader A practical Douyin downloader for both single-item and profile batch downloads, with progress display, retries, SQLite deduplication, and browser fallback support. 抖…...

CKA-2026-resources

您管理一个 WordPress 应用程序。由于资源请求过高,某些 Pod 无法启动。Taskrelative-fawn namespace 中的 WordPress 应用程序包含:l具有 3 个副本的 WordPress Deployment按如下方式调整所有 Pod 资源请求:l将节点资源平均分配给这 3 个 Po…...

CLIP-GmP-ViT-L-14模型蒸馏实战:基于STM32F103C8T6的轻量化部署探索

CLIP-GmP-ViT-L-14模型蒸馏实战:基于STM32F103C8T6的轻量化部署探索 1. 引言 想象一下,一个只有指甲盖大小、成本低廉的微控制器,能够理解一张图片和一段文字是否匹配。这听起来像是科幻电影里的场景,但今天,我们就要…...

【世纪龙科技】3D仿真还原真车,拆装检测步步有方

新能源汽车动力总成拆装与检测虚拟实训软件—— 虚实相融,赋能未来工匠的成长新范式在新能源汽车产业蓬勃发展的今天,职业院校作为技术技能人才的摇篮,正面临着“高压安全难保障、精密部件难拆装、大班教学难兼顾”的实训新挑战。如何让学生在…...

如何在 PHP 包含文件中动态排除当前页面对应的导航项

本文介绍如何通过 PHP 动态控制 include() 的执行时机,实现在侧边栏(如 aside.php)中自动隐藏当前页面对应的导航链接,无需额外语言或框架,纯 PHP 即可实现。 本文介绍如何通过 php 动态控制 include() 的执行时机…...

Go语言怎么防SQL注入_Go语言SQL注入防护教程【深入】

必须使用参数占位符(如?或$1)而非字符串拼接来防止SQL注入;sql.RawBytes仅用于读取二进制字段,不可用于拼接SQL;动态表名/字段名需白名单校验;ORM应禁用Raw()并启用PrepareStmt;JSON中的SQL片段…...

知识的基本特性:相对正确性、不确定性与可表示性

“知识”并不是对客观世界的简单照搬,也不是永远不变的绝对真理。它是在认识、概括、组织和应用过程中形成的结果,因此既具有稳定性,也具有条件性。理解知识的基本特性,有助于进一步理解:为什么知识需要表示&#xff0…...

语义网络表示法:从节点、关系到继承推理

在知识表示的发展过程中,语义网络表示法(Semantic Network Representation)是一种非常重要的方法。它用“节点—关系—节点”的结构来表示知识,把对象及其联系组织成有向图,因此比单纯的逻辑公式更直观,也更…...

Wand-Enhancer:3分钟解锁WeMod专业功能的终极指南

Wand-Enhancer:3分钟解锁WeMod专业功能的终极指南 【免费下载链接】Wand-Enhancer Advanced UX and interoperability extension for Wand (WeMod) app 项目地址: https://gitcode.com/gh_mirrors/we/Wand-Enhancer 还在为WeMod的专业功能限制而烦恼吗&#…...

如何在Windows 11上运行Android应用:Windows Subsystem for Android完整指南

如何在Windows 11上运行Android应用:Windows Subsystem for Android完整指南 【免费下载链接】WSA Developer-related issues and feature requests for Windows Subsystem for Android 项目地址: https://gitcode.com/gh_mirrors/ws/WSA Windows Subsystem …...

零代码:CAM++说话人识别系统,可视化界面完成语音比对

零代码:CAM说话人识别系统,可视化界面完成语音比对 1. 系统概述 CAM说话人识别系统是一款基于深度学习的声纹识别工具,通过直观的可视化界面让用户无需编写代码即可完成语音比对和特征提取。该系统由开发者"科哥"基于阿里达摩院开…...

Phi-4-mini-reasoning 3.8B在VSCode中的智能编程应用:Codex风格体验

Phi-4-mini-reasoning 3.8B在VSCode中的智能编程应用:Codex风格体验 1. 轻量级AI编程助手的惊艳表现 在编程领域,AI辅助工具正变得越来越重要。Phi-4-mini-reasoning 3.8B作为一款轻量级模型,在VSCode中展现出了令人惊喜的智能编程能力。虽…...

第十六届 蓝桥杯嵌入式设计与开发 省赛 客观题

不定项选择,共10题 01.关于STM32时钟源的说法,错误的是() A.HSI精度高于HSE B.LSE常用于RTC模块 C.PLL可将外部或内部时钟倍频 D.切换系统时钟源或修改主频时,必须先进入停机模式 答案:AD A:HSI(内部高速时钟&#xff…...

文墨共鸣大模型Dify平台无缝集成:可视化构建AI文本处理应用

文墨共鸣大模型Dify平台无缝集成:可视化构建AI文本处理应用 你是不是也遇到过这样的场景:手头有一个很棒的AI大模型,比如文墨共鸣,但每次想用它做点事情,都得写代码、调接口,过程繁琐,门槛不低…...

macOS 强制运行拦截程序

当你从 Chrome、Safari 或其它网络渠道下载文件时,macOS 会自动给这个文件贴上一张“隐形贴纸”,名字就叫 com.apple.quarantine。系统的逻辑: 当你双击运行一个文件时,系统的 Gatekeeper会先检查有没有这张贴纸。拦截逻辑&#x…...

实测Qwen3智能字幕生成效果:高精度时间戳对齐,剪辑无缝衔接

实测Qwen3智能字幕生成效果:高精度时间戳对齐,剪辑无缝衔接 1. 效果展示与核心价值 1.1 为什么选择Qwen3字幕生成工具 在视频制作过程中,字幕时间轴对齐是最耗时的工作之一。传统手动对齐方式不仅效率低下,而且很难达到毫秒级精…...

终极显卡驱动清理指南:DDU工具完整使用教程

终极显卡驱动清理指南:DDU工具完整使用教程 【免费下载链接】display-drivers-uninstaller Display Driver Uninstaller (DDU) a driver removal utility / cleaner utility 项目地址: https://gitcode.com/gh_mirrors/di/display-drivers-uninstaller Displ…...

Sunshine游戏串流服务器:5步搭建你的专属云端游戏平台

Sunshine游戏串流服务器:5步搭建你的专属云端游戏平台 【免费下载链接】Sunshine Self-hosted game stream host for Moonlight. 项目地址: https://gitcode.com/GitHub_Trending/su/Sunshine 想要在任何设备上畅玩PC游戏大作,却受限于硬件配置&a…...

Qwen2.5-VL-7B-Instruct部署教程:GPU算力监控(nvidia-smi)+服务健康检查脚本

Qwen2.5-VL-7B-Instruct部署教程:GPU算力监控(nvidia-smi)服务健康检查脚本 1. 项目概述 Qwen2.5-VL-7B-Instruct是一款强大的多模态视觉-语言模型,能够同时处理图像和文本输入,生成高质量的响应。该模型特别适合需要…...

A-47 矿山井下通信应用

矿山井下属于高噪声、强回声、长巷道、多干扰、潮湿粉尘恶劣环境,传统对讲、扩音、拾音设备普遍存在人声被机械噪音淹没、回声啸叫严重、通话卡顿失真、远距离拾音困难、电磁干扰杂音大等问题,严重影响安全生产调度与应急救援通信。A-47 模块集成AEC 回音…...

UnrealPakViewer终极指南:如何快速分析虚幻引擎Pak文件资源

UnrealPakViewer终极指南:如何快速分析虚幻引擎Pak文件资源 【免费下载链接】UnrealPakViewer 查看 UE4 Pak 文件的图形化工具,支持 UE4 pak/ucas 文件 项目地址: https://gitcode.com/gh_mirrors/un/UnrealPakViewer 你是否曾经面对数十GB的虚幻…...

大语言模型作为语种民族文明压缩镜像的映射特性分析

摘要 大语言模型通过预测下一个词学习语言概率模式的本质,使其成为其所训练语料库的统计压缩体。这种本质决定了模型能够映射特定语种民族或文明的深层文化偏好,成为一个独特的“压缩镜像”。该镜像并非对文明的完整复制,而是基于海量文本数据…...

5分钟掌握SketchUp STL插件:从3D建模到3D打印的完整转换指南

5分钟掌握SketchUp STL插件:从3D建模到3D打印的完整转换指南 【免费下载链接】sketchup-stl A SketchUp Ruby Extension that adds STL (STereoLithography) file format import and export. 项目地址: https://gitcode.com/gh_mirrors/sk/sketchup-stl 你是…...

CogVideoX-2b镜像避坑指南:解决显存溢出、黑屏等常见问题

CogVideoX-2b镜像避坑指南:解决显存溢出、黑屏等常见问题 1. 为什么你需要这份避坑指南 当你第一次尝试使用CogVideoX-2b生成视频时,可能会遇到各种意外情况:显存突然爆满、生成的视频全是黑屏、或者等待了十分钟却没有任何输出。这些问题不…...

Star CCM+ 实战:旋风分离器(cyclone separator)体网格生成与优化策略

1. 旋风分离器网格生成前的准备工作 在开始使用Star CCM生成旋风分离器体网格之前,我们需要做好充分的准备工作。旋风分离器作为一种常见的气固分离设备,其内部流动特性复杂,包含强烈的旋转流场和湍流现象。这就对网格质量提出了更高要求&am…...

深度掌控AMD Ryzen:SMUDebugTool硬件级调试全攻略

深度掌控AMD Ryzen:SMUDebugTool硬件级调试全攻略 【免费下载链接】SMUDebugTool A dedicated tool to help write/read various parameters of Ryzen-based systems, such as manual overclock, SMU, PCI, CPUID, MSR and Power Table. 项目地址: https://gitcod…...

五年磨剑与二十年深耕:5 年与 20 年程序员的差距,远不止代码本身

在信息技术飞速迭代的今天,程序员这一职业始终站在时代前沿。有人说,程序员是吃 “青春饭” 的行业,年轻意味着精力充沛、学习速度快、能熬夜加班;也有人说,真正的技术高手,往往藏在十几年甚至二十余年的行…...

解锁Steam游戏新体验:开源成就管理工具深度解析

解锁Steam游戏新体验:开源成就管理工具深度解析 【免费下载链接】SteamAchievementManager A manager for game achievements in Steam. 项目地址: https://gitcode.com/gh_mirrors/st/SteamAchievementManager 你是否曾因为一个难以获得的成就而反复尝试同一…...