二十三、图的遍历(BFS和DFS)
一、概念
图的遍历(Traversing Graph)从某一顶点出发,访问图中所有顶点,且使每一顶点仅被访问一次。与树的遍历不同的是,图的遍历需要处理两种特殊情况:一是从某一顶点出发进行遍历时,可能访问不到所有其他顶点,比如非连通图;二是有些图存在回路,必须保证遍历过程不能因为回路陷入死循环。
图的遍历是解决图的许多应用问题的基础,如路径问题、连通性问题等。图的遍历有两种基本方法:
- 深度优先遍历(Depth-First Search, DFS)
- 广度优先遍历(Breadth-First Search, BFS)
二、深度优先遍历
深度优先遍历,类似于树的前序遍历的过程,以“深度”作为第一关键词,沿着一条路径直到无法继续前进,才退回到路径上离当前顶点最近的还存在未访问分支顶点的岔道口,并前往访问那些未访问分支顶点,直到遍历完整个图。
过程如下:
- 从图中某个顶点 $v$ 出发,访问 $v$ 。
- 找出刚访问过的顶点的第一个未被访问的邻接点,访问该顶点。以该顶点为新顶点,重复此步骤,直至刚访问过的顶点,没有未被访问的邻接点为止。
- 返回前一个访问过的且仍有未被访问的邻接点的顶点,找出该顶点的下一个未被访问的邻接点,访问该顶点。
- 重复步骤2和3,直至图中所有顶点都被访问过,搜索结束。
DFS的具体实现
DFS遍历图的基本思路就是将经过的顶点设置为已访问,在下次递归碰到这个顶点的时就不再去处理,直到整个图的顶点都被标记为已访问。
动图演示

邻接矩阵
1.连通图的深度优先遍历
Status DFS_M(MGraph G, int k, Status(*visit)(int))
{ //从连通图G的k顶点出发进行深度优先遍历,图G采用邻接数组存储结构
int i;
if (visit(k) == ERROR) //访问 k 顶点
return ERROR;
G.tags[k] = VISITED;
for (i = FirstAdjVex_M(G, k); i >= 0; i = NextAdjVex_M(G, k, i))
{
if (G.tags[i] == UNVISITED) //位序为i的邻接顶点未被访问过
if (DFS_M(G, i, visit) == ERROR) //对 i 顶点递归深度遍历
return ERROR;
}
return OK;
}
2.图的深度优先遍历
Status DFSTraverse_M(MGraph G, Status(*visit)(int))
{ //深度优先遍历采用邻接数组存储结构的图G
int i;
for (i = 0; i < G.n; i++)
G.tags[i] = UNVISITED; //初始化标志数组
for (i = 0; i < G.n; i++)
if (G.tags[i] == UNVISITED) //若i顶点未访问,则以其为起点进行深度优先遍历
if (DFS_M(G, i, visit) == ERROR)
return ERROR;
return OK;
}
深度优先搜索遍历的算法分析
在遍历图时,对图中每个顶点至多调用一次DFS函数,因为一旦某个顶点被标志成已被访问,就不再从它出发进行搜索。因此,遍历图的过程实质上是对每个顶点查找其邻接点的过程,其耗费的时间则取决于所采用的存储结构。
- 当用邻接矩阵表示图时,查找每个顶点的邻接点的时间复杂度为 $O({n^2})$,其中 $n$ 为图中顶点数。
- 当以邻接表做图的存储结构时,查找邻接点的时间复杂度为 $O(e)$,其中 $e$ 为图中边数。由此,当以邻接表做存储结构时,深度优先搜索遍历图的时间复杂度为 $O(n+e)$。
三、广度优先遍历
广度优先遍历,类似于树的按层次遍历的过程,以“广度”作为关键词,每次以扩散的方式向外访问顶点。
过程如下:
- 从图中某个顶点 $v$ 出发,访问 $v$ 。
- 依次访问 $v$ 的各个未曾访问过的邻接点。
- 分别从这些邻接点出发依次访问它们的邻接点,并使“先被访问的顶点的邻接点”先于“后被访问的顶点的邻接点”被访问。重复步骤3,直至图中所有已被访问的顶点的邻接点都被访问到。
BFS的具体实现
使用BFS遍历图的基本思想是建立一个队列,并把初始顶点加入队列,此后每次都取出队首顶点进行访问,并把从该顶点出发可以到达的未曾加入过队列(而不是未访问)的顶点全部加入队列,直到队列为空。
动图演示

邻接表
1.图的广度优先遍历
Status BFSTraverse_AL(ALGraph G, Status(*visit) (int))
{ //广度优先遍历采用邻接表存储结构的图G
int i, j, k;
AdjVexNodeP p;
LQueue Q;
InitQueue_LQ(Q); //初始化链队列Q
for (i = 0; i < G.n; i++)
G.tags[i] = UNVISITED; //初始化标志数组
for (i = 0; i < G.n; i++) //依次检查所有顶点
if (G.tags[i] == UNVISITED) // i 顶点未被访问
{
if (visit(i) == ERROR)
return ERROR;
G.tags[i] = VISITED;
EnQueue_LQ(Q, i); //访问 i 顶点,并入队
while (DeQueue_LQ(Q, k) == OK) //出队元素到 k
{
for (j = FirstAdjVex_AL(G, k, p); j >= 0; j = NextAdjVex_AL(G, k, p))
//依次判断 k 顶点的所有邻接顶点j, 若未曾访问,则访问它,并入队
{
if (G.tags[j] == UNVISITED)
{
if (visit(j) == ERROR)
return ERROR;
G.tags[j] = VISITED;
EnQueue_LQ(Q, j);
}
}
}
}
return OK;
}
广度优先搜索遍历的算法分析
由算法可知,每个顶点至多进一次队列。遍历图的过程实质上是通过边找邻接点的过程, 因此广度优先搜索遍历图的时间复杂度和深度优先搜索遍历相同。两种遍历方法的不同之处仅仅在于对顶点访问的顺序不同。
- 当用邻接矩阵表示图时,查找每个顶点的邻接点的时间复杂度为 $O({n^2})$,其中 $n$ 为图中顶点数。
- 当以邻接表做图的存储结构时,广度优先搜索遍历图的时间复杂度为 $O(n+e)$。
二十三、图的遍历(BFS和DFS)的更多相关文章
- 图的遍历BFS广度优先搜索
图的遍历BFS广度优先搜索 1. 简介 BFS(Breadth First Search,广度优先搜索,又名宽度优先搜索),与深度优先算法在一个结点"死磕到底"的思维不同,广度优先 ...
- 图的遍历BFS
图的遍历BFS 广度优先遍历 深度优先遍历 可以进行标记 树的广度优先遍历,我们用了辅助的队列 bool visited[MAX_VERTEX_NUM] //访问标记数组 //广度优先遍历 void ...
- 数据结构(三十二)图的遍历(DFS、BFS)
图的遍历和树的遍历类似.图的遍历是指从图中的某个顶点出发,对图中的所有顶点访问且仅访问一次的过程.通常有两种遍历次序方案:深度优先遍历和广度优先遍历. 一.深度优先遍历 深度优先遍历(Depth_Fi ...
- 图的遍历算法:DFS、BFS
在图的基本算法中,最初需要接触的就是图的遍历算法,根据访问节点的顺序,可分为深度优先搜索(DFS)和广度优先搜索(BFS). DFS(深度优先搜索)算法 Depth-First-Search 深度优先 ...
- Kruskal和prime算法的类实现,图的遍历BFS算法。
一.图的遍历 #include<iostream> #include<queue> #include<vector> using namespace std; in ...
- 图的遍历——BFS
原创 裸一篇图的BFS遍历,直接来图: 简单介绍一下BFS遍历的过程: 以上图为例子,从0开始遍历,访问0,按大小顺序访问与0相邻的所有顶点,即先访问1,再访问2: 至此顶点0已经没有作用了,因为其本 ...
- Java实现 洛谷 P3916 图的遍历(反向DFS+记忆化搜索)
P3916 图的遍历 输入输出样例 输入 4 3 1 2 2 4 4 3 输出 4 4 3 4 import java.io.BufferedReader; import java.io.IOExce ...
- 模板 图的遍历 bfs+dfs 图的最短路径 Floyed+Dijkstra
广搜 bfs //bfs #include<iostream> #include<cstdio> using namespace std; ],top=,end=; ][]; ...
- 《算法设计手册》面试题解答 第五章:图的遍历 附:DFS应用之找挂接点
第五章面试题解答 5-31. DFS和BFS使用了哪些数据结构? 解析: 其实刚读完这一章,我一开始想到的是用邻接表来表示图,但其实用邻接矩阵也能实现啊?后来才发现应该回答,BFS用队列实现:DFS可 ...
- 图的遍历——BFS(队列实现)
#include <iostream> #include <cstdio> #include <cstdlib> #include <cstring> ...
随机推荐
- Android掌控WiFi不完全指南
前言 如果想要对针对WiFi的攻击进行监测,就需要定期获取WiFi的运行状态,例如WiFi的SSID,WiFi强度,是否开放,加密方式等信息,在Android中通过WiFiManager来实现 WiF ...
- Silky微服务框架之模块
模块的定义 Silky是一个包括多个nuget包构成的模块化的框架,每个模块将程序划分为一个个小的结构,在这个结构中有着自己的逻辑代码和自己的作用域,不会影响到其他的结构. 模块类 一般地,一个模块的 ...
- 每日算法3:随机生成五个不同整数,将数字转换为RMB格式
随机生成五个不同整数 点击查看代码 /* 题目解析: 1.采用Math对象的random()方法, 2.将每次生成的数跟之前的数判断相等则此次生成无效i-- */ function randomNum ...
- Java多线程(7):JUC(上)
您好,我是湘王,这是我的博客园,欢迎您来,欢迎您再来- 前面把线程相关的生命周期.关键字.线程池(ThreadPool).ThreadLocal.CAS.锁和AQS都讲完了,现在就剩下怎么来用多线程了 ...
- 如何判断多个url的状态 即是否能成功访问?
假设我们所有的url在excel种我们可以用loadwork 这个库 遍历所有的url 放入 列表 再通过request 或者 httpx 来判断 这个网址status_code 进而放入mongo种 ...
- Flask框架:如何运用Ajax轮询动态绘图
摘要:Ajax是异步JavaScript和XML可用于前后端交互. 本文分享自华为云社区<Flask框架:运用Ajax轮询动态绘图>,作者:LyShark. Ajax是异步JavaScri ...
- 真正“搞”懂HTTP协议05之What's HTTP?
前面几篇文章,我从纵向的空间到横向的时间,再到一个具体的小栗子,可以说是全方位,无死角的覆盖了HTTP的大部分基本框架,但是我聊的都太宽泛了,很多内容都是一笔带过,再加上一句后面再说就草草结束了.并且 ...
- ValueError: Detected newline in header value. This is a potential security problem
原因 flask框架进行重定向的url中包含 换行符\n或\r 解决方法 使用 strip() 函数去除行首或行尾的换行符(如果你url中间包含这些符号replace函数替换, 但是如果中间包含只能说 ...
- MySQL DATE_SUB查询工龄大于35的员工信息
#(11) 查询工龄大于或等于35年的员工信息.SELECT * FROM emp e WHERE e.HIREDATE<=DATE_SUB(SYSDATE(),INTERVAL 35 YEAR ...
- tostring、(string)和 String.valueOf()
上周遇到一个问题,只怪自己平时没注意这个细节,从数据库取数据在map集合里,取出该值是我用了.tostring的方法,一次在当取出数据为空时代码报java.lang.NullPointerExcept ...