图详解第二篇:图的遍历(广度优先+深度优先)

这篇具有很好参考价值的文章主要介绍了图详解第二篇:图的遍历(广度优先+深度优先)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

图的遍历

所谓图的遍历:

即从图中的任一顶点出发,对图中的所有顶点访问一次且只访问一次。
给定一个图G和其中任意一个顶点v0,从v0出发,沿着图中各边访问图中的所有顶点,且每个顶点仅被遍历一次。

ps:

我们后面讲解这些图相关的算法默认都针对邻接矩阵结构的图去讲解,因为后面有些算法针对的图一般都是比较稠密的图,前面我们说了邻接矩阵更适合稠密图。

那具体要如何对一个图进行遍历呢?有哪些方法呢?

1. 图的广度优先遍历(一石激起千层浪)

什么是广度优先遍历呢?

图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
其实我们之前学过的二叉树的层序遍历就是一种广度优先遍历,要借助一个队列来搞,下面对图的广度优先遍历也是一样

思路分析

那图的广度优先遍历是怎么样的呢?举个栗子:

图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
这就是对一个图(无向图)的广度优先遍历,红色的数字就是结点遍历的顺序。
大家看,其实就是一层一层的遍历嘛,这里是从A这个顶点开始,所以先遍历结点A,然后依次遍历与A直接相连的一层的结点,接着逐层向外扩展,直到遍历完所有可达的节点。

那我们用代码要怎么实现呢?

其实就跟二叉树的层序遍历类似,借助一个队列就可以搞定。
比如就以这个图为例:
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
从顶点A开始,那就让A先入队列。
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
队列不为空,出队头元素A,然后我们打印一下A的值,那这个顶点就遍历过去了,然后把与A直接相连的顶点BCD入队列
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
队列不为空,继续出队头元素B,然后把与B直接相连的顶点入队列

那现在就出现了一个问题:

现在要把与B直接相连的顶点入队列,而与B直接相连的顶点是A、C、E。
但是,A前面已经遍历过了,C现在也在队列里面呢!
E入队列的话是没问题的,他就是下一层的,但是AC要是再入队列的话这就不对了啊。

那我们如何解决这个问题呢?如果避免某些顶点会重复入队列呢?

🆗,我们可以想办法对已经遍历过的结点做一个标记,这个即使后面再遇到,我们也能分辨出来,不让它再入队列了。

那如何标记已经遍历过的顶点呢?

每个顶点不是都映射一个下标嘛。
所以我们就可以开一个数组,默认都给false,遍历一个结点,就把对应下标位置的值改为true,表示这个结点已经被遍历过了。

那我们何时去标记一个顶点呢?打印之后标记还是入队列就标记呢?

如果打印之后标记的话,B出队列之后其实还会把C带到队列里面
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
因为B出队列,然后打印B,此时A已经打印过了被标记了,但是C还没有出队列打印,所以C还没有被标记,所以B打印之后入与B直接相连的顶点ACE的时候,A不会入,但是C还会入。
不管也没关系,因为出到第二个C的时候,第一个C 肯定被标记了,那我们就可以判断,然后不打印第二个C,只把它出队列。

当然更好的做法是这样的:

一个顶点入队列的时候我们就去标记它,这样就不会出现上面的情况(队列里面出现重复顶点)

代码实现

那我们来写一下代码:

图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
调用的时候给一个起始顶点就可以了
那我们需要一个队列和一个标记数组,首先起始顶点入队列
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
然后我们就按照上面的逻辑一层一层走就行了,循环出队列里面的元素并把与之直接相连的结点入队列,直到队列为空就是遍历完了,循环结束
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图

🆗,就写好了

测试

那我们来测试一下:

图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
它对应的图应该是这样的
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
我们现在以张三为起点对其进行广度优先遍历
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
我们来看下结果对不对
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
没问题

美团2020校招笔试题:六度人脉

然后我们来看一道美团曾经考过的一道问答题:
题目链接: link
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
大家自己先读一下题

🆗,我们来分析一下:

这道题目最终的问题是让我们为小点推荐6度好友,如何找到哪些是它的6度好友呢?
其实很简单:
如果我们用上面实现的广度优先遍历去遍历对应的好友的关系图的话,我们一层一层的打印,第六层打印出来的人就是6度好友

那我们上面实现的广度优先遍历打印的时候并没有分层打印,所以我们可以给上面的代码优化一下,让它分层打印就行了:

那如何做到分层打印呢?
其实这个问题我们之前也有遇到过,前面的文章里我们有讲过一个OJ题,也是二叉树的层序遍历,但是要求把每一层的结点都分开存放到一个数组里面,最终返回的就是一个二维数组嘛。
大家回想一下我们当时怎么做的?
🆗,增加一个变量,去记录每一层结点的个数,每出一个,就- -一次,一层出完,继续记录下一层的个数,这样就可以控制分层打印了。
那我们来把上面的代码改进一下
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图

试一下:

图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
就实现分层了

2. 图的深度优先遍历(一条道走到黑)

那深度优先遍历又是什么呢?

图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
其实我们之前学的二叉树的前序遍历就是一个深度优先遍历,就是先往深了去走,直到走不通了,再往回回溯。
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图

思路分析

那图的深度优先遍历我们来看一下:

图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
大家看这个图。
起点是A,那就从A开始,A遍历完,找一个与它直接相连的且没被遍历过的(所以我们这里还要标记遍历过的结点),那我们这里用的图结构是邻接矩阵的话,他找相连顶点的时候肯定就是按照那个结点对应的下标从小到大去找嘛。
那A找到的是B,然后B再去找一个与它邻接的顶点遍历(广度的话B遍历完就是继续找其它与A邻接的遍历),那B找到了C,那后面同样的,C再去找,接着…
一直走,一直走;
直到找到某一个顶点它的所有邻接顶点都被遍历过了,然后往回退,再去走其它没有走过的路径去遍历

所以我们还是用递归来搞就很简单嘛(先递推,再回归),DFS一般都是用递归来实现的。

代码实现

我们来写一下代码:

图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
那这里递归的话我们可以在搞一个子函数,这基本上算固定套路了
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
那这里递归的时候我们传两个参数,首先是本次递归的起点下标(因为底层是邻接矩阵嘛),然后第二个是标记数组,这个要一直往下传,并且上一层递归的改变要延续到下一层,所以要传引用。不传这个的话你就搞一个全局的标记数组。
那递归里面的逻辑呢,很简单:
先访问当前顶点,然后找一个没访问过的邻接顶点,继续递归往深走,一直走到走不动了自动就向上返回了
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图

就写好了

测试

我们来测试一下:

就还用上面那个图吧
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
运行一下
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
是没问题的,大家可以自己验证一下
图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图

3. 对于非连通图情况的处理

上一篇文章我们有讲过连通图的概念:

那其实我们上面演示的例子,都是用的连通图。
对于连通图,不论是广度优先遍历还是深度优先遍历,我们上面的代码肯定都是没问题的,肯定能遍历完所有的顶点;

但是如果给我们的图是一个非连通的图呢?比如:

图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图
这样的。
那我们上面的代码针对这种情况会出现什么问题?
对于这种非连通图,如果我们再用上面的代码进行遍历的话,任取一个结点作为起点,比如还是A
那最终的结果就是我们只能遍历到上面的6个结点,最终遍历结束,还剩下面的3个结点我们是遍历不到的,因为它们跟上面的不连通,根本走不下来。

那我们如何解决这个问题呢?

其实也好办:
可再在搞一个循环,每遍历一次之后,我们就去那个标记数组里面看还有没有没被遍历到的顶点,如果有的话,就再取一个没被遍历到的点作为起点,再进行对应的DFS/BFS遍历。
直到标记数组里面所有的位置都变成true,那就证明所有的顶点都被遍历过了。
至此,图的遍历真正结束!

这样就可以解决非连通图的遍历问题!

4.源码

BFS

void BFS(const V& src)
{
	size_t srci = GetVertexIndex(src);
	//队列
	queue<int> q;
	//标记数组
	vector<bool> visited(_vertexs.size(), false);

	q.push(srci);
	visited[srci] = true;
	int levelSize = 1;

	while (!q.empty())
	{
		//每次循环打印一层
		while (levelSize--)
		{
			//取对头顶点front并打印
			int front = q.front();
			q.pop();
			cout << front << ":" << _vertexs[front] << " ";

			//把front顶点的邻接顶点入队列
			for (size_t i = 0; i < _vertexs.size(); i++)
			{
				//过滤掉已经入过队列的
				if (_matrix[front][i] != MAX_W && visited[i] == false)
				{
					q.push(i);
					visited[i] = true;
				}
			}
		}
		cout << endl;
		//更新levelSize为下一层的结点个数
		levelSize = q.size();
	}
	cout << endl;
}

DFS

void _DFS(size_t srci, vector<bool>& visited)
{
	//遍历当前结点并标记
	cout << srci << ":" << _vertexs[srci] << endl;
	visited[srci] = true;

	//寻找当前结点没被访问过的邻接顶点,继续递归往深度遍历
	for (size_t i = 0; i < _vertexs.size(); i++)
	{
		if (_matrix[srci][i] != MAX_W && visited[i] == false)
		{
			_DFS(i, visited);
		}
	}
}
void DFS(const V& src)
{
	size_t srci = GetVertexIndex(src);
	//标记数组
	vector<bool> visited(_vertexs.size(), false);

	_DFS(srci, visited);
}

图结构我们用的是邻接矩阵,需要代码大家可以去上一篇文章获取…

图详解第二篇:图的遍历(广度优先+深度优先),高阶数据结构(C++),数据结构,宽度优先,深度优先,图文章来源地址https://www.toymoban.com/news/detail-715565.html

到了这里,关于图详解第二篇:图的遍历(广度优先+深度优先)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 图的遍历(广度优先遍历BFS,深度优先遍历DFS)

    目录 图的遍历概念: 图的广度优先遍历(BFS): 代码实现如下: 测试如下: 注意: 图的深度优先遍历(DFS): 代码实现如下: 测试如下: 总代码: 结语: 给定一个图G和其中任意一个顶点v0,从v0出发,沿着图中各边访问图中的所有顶点,且每个顶点仅被遍历一次。\\\"遍

    2024年02月21日
    浏览(56)
  • 图的遍历-深度优先遍历与广度优先遍历(C语言)

    图的遍历 概念:指的是从图中的任一顶点出发,对图中的所有顶点访问一次且只访问一次。 邻接矩阵及邻接表的创建 : 图的存储结构-无向邻接矩阵与无向邻接表(C语言). 结构定义 邻接矩阵的深度优先遍历操作 邻接矩阵的深度优先递归算法 结构定义 邻接表的深度优先遍

    2024年02月06日
    浏览(48)
  • 大话数据结构-图的深度优先遍历和广度优先遍历

      图的遍历分为深度优先遍历和广度优先遍历两种。   深度优先遍历(Depth First Search),也称为深度优先搜索,简称DFS,深度优先遍历,是指从某一个顶点开始,按照一定的规则,访问并记录下一个未访问顶点。对于非连通图,则是按连通分量,采用同一规则进行深度优

    2024年02月04日
    浏览(58)
  • 图的遍历之 深度优先搜索和广度优先搜索

    深度优先搜索的图文介绍 1. 深度优先搜索介绍 图的深度优先搜索(Depth First Search),和树的先序遍历比较类似。 它的思想:假设初始状态是图中所有顶点均未被访问,则从某个顶点v出发,首先访问该顶点,然后依次从它的各个未被访问的邻接点出发深度优先搜索遍历图,直至

    2024年02月13日
    浏览(53)
  • 蛮力算法之深度优先遍历和广度优先遍历——图的深度优先遍历和广度优先遍历,附带案例:迷宫问题及矩阵中传染性传播问题

    这两种搜索方法本质上都是基于蛮力法思路 这两种搜索方法对有向图和无向图都适用 1.1 邻接矩阵 邻接矩阵示意图: 1.2 邻接表 注意:边结点代表的是图中的边,而并非图中的结点;头结点才表示图中的结点;头结点身后所连接的为边结点 邻接表示意图:(一般默认为出边

    2024年02月05日
    浏览(66)
  • (超详细)C++图的深度优先遍历、广度优先遍历(数据结构)

            根据下图,编写代码实现图的深度优先遍历和广度优先遍历。          按照英文字母顺序,以邻接表为存储结构,实现图的深度优先和广度优先遍历。遍历的顺序从顶点a开始。 以用户指定的结点为起点,分别输出每种遍历下的结点访问序列。   (1)从顶点a,

    2024年02月08日
    浏览(61)
  • 【数据结构与算法】图的深度优先和广度优先遍历

    😊😊作者简介😊😊 : 大家好,我是南瓜籽,一个在校大二学生,我将会持续分享Java相关知识。 🎉🎉个人主页🎉🎉 : 南瓜籽的主页 ✨✨座右铭✨✨ : 坚持到底,决不放弃,是成功的保证,只要你不放弃,你就有机会,只要放弃的人,他肯定是不会成功的人。 图是一

    2024年02月02日
    浏览(58)
  • 【数据结构】图的遍历:广度优先(BFS),深度优先(DFS)

    目录 1、广度优先(BFS) 算法思想  广度优先生成树  知识树  代码实现  2、深度优先(DFS) 算法思想  深度优先生成树 知识树  代码实现           图的广度优先遍历(BFS)是一种遍历图的算法,其思想是从起始顶点开始遍历图,先访问起始顶点的所有直接邻居,然

    2024年02月04日
    浏览(66)
  • 数据结构学习笔记——图的遍历算法(深度优先搜索和广度优先搜索)

    图的遍历指从图中某一顶点出发(任意一个顶点都可以作为访问的起始顶点),按照某种遍历方法,对图中所有的顶点访问一次且只访问一次。图与树不一样,其中一个顶点可能与多个顶点相连,所以需记录已访问过的顶点,当访问一个顶点后,考虑如何选取下一个要访问的

    2024年02月05日
    浏览(53)
  • 图的遍历(搜索)算法(深度优先算法DFS和广度优先算法BFS)

    从图的某个顶点出发访问遍图中所有顶点,且每个顶点仅被访问一次。(连通图与非连通图) 1、访问指定的起始顶点; 2、若当前访问的顶点的邻接顶点有未被访问的,则任选一个访问之;反之,退回到最近访问过的顶点;直到与起始顶点相通的全部顶点都访问完毕; 3、若

    2024年01月17日
    浏览(49)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包