【数据结构】图的存储与遍历

这篇具有很好参考价值的文章主要介绍了【数据结构】图的存储与遍历。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

图的概念

图是由顶点集合及顶点间的关系组成的一种数据结构:G = (V, E)

图分为有向图和无向图

  • 在有向图中,顶点对<x, y>是有序的,顶点对<x,y>称为顶点x到顶点y的一条边(弧),<x, y>和<y, x>是两条不同的边。
  • 在无向图中,顶点对(x, y)是无序的,顶点对(x,y)称为顶点x和顶点y相关联的一条边,这条边没有特定方向,(x, y)和(y,x)是同一条边


【数据结构】图的存储与遍历,数据结构,数据结构,算法,图论

就像第一幅图中,<V1,V2>顶点构成有向图,边 E1和边E2是不同指向的

在第二幅图中,<V3,V4>构成无向图,E3就是连接二者关系的唯一边。

在生活中的关系,例如微信里的朋友关系就是无向的,只有双方都相连,才能发送消息

而类是与微博等就是单向的,你关注某人,就是一条边。当他也关注你时,才构成俩条边。

 

完全图

  • 有n个顶点的无向图中,若有n * (n-1)/2条边,即任意两个顶点之间有且仅有一条边,
  • 则称此图为无向完全图
  • 在n个顶点的有向图中,若有n * (n-1)条边,即任意两个顶点之间有且仅有方向相反的边,则称此图为有向完全图

下列的图都是完全图 

【数据结构】图的存储与遍历,数据结构,数据结构,算法,图论

 

邻接顶点

  • 无向图中,v-u 称互为邻接顶点
  • 有向图中,v->u   称u是v的邻接顶点

与顶点直接相邻的顶点

例如:G1中 0和2 都是 1的邻接顶点

0 1 2 3互为邻接顶点

【数据结构】图的存储与遍历,数据结构,数据结构,算法,图论

图与树的主要联系与区别

树是一种特殊的图

图不一定是树

树关注结点的存值,图关注顶点和边的权值。


图的存储结构

本质就是将边和顶点,及其关系存储起来。

用vector数组保存顶点

vector<V> _vertexs;

利用map映射顶点和下标的关系

		map<V, int> _vIndexMap;	

为了解决俩个顶点是否相连,相连的边权值是多少的问题。

解决方法主要有邻接矩阵,和邻接表


邻接矩阵

利用一个N*N的矩阵表示边和权值的关系

【数据结构】图的存储与遍历,数据结构,数据结构,算法,图论

如果想要找到顶点A相连的顶点,通过横行找到A,在通过纵行 找到 内容不为无穷大的 B D。

为了表示边权之间的关系,修改 0和1为权值w和无穷

规定顶点自己到自己是不相连 ,不连通为无穷。

【数据结构】图的存储与遍历,数据结构,数据结构,算法,图论

 

namespace Matrix
{
	template<class V, class W, W MAX_W = INT_MAX, bool Direction = false>
	class Graph
	{
		typedef Graph<V, W, MAX_W, Direction> Self;
	public:
		Graph()
		{

		}
		Graph(const V* a, size_t n)
		{
			_vertexs.reserve(n);
			for (size_t i = 0; i < n; i++)
			{
				_vertexs.push_back(a[i]);
				_vIndexMap[a[i]] = i;
			}
			_matrix.resize(n);
			for (size_t i = 0; i < _matrix.size(); i++)
			{
				_matrix[i].resize(n, MAX_W);
			}
		}

		//获取下标
		size_t GetVertexIndex(const V& v)
		{
			auto it = _vIndexMap.find(v);
			if (it != _vIndexMap.end())
			{
				return it->second;
			}
			else
			{
				throw invalid_argument("顶点不存在");
				assert(false);
				return -1;
			}
		}

		void _AddEdge(size_t srci, size_t dsti, const W& w)
		{
			_matrix[srci][dsti] = w;
			
			if (Direction == false)
			{
				_matrix[dsti][srci] = w;
			}
		}

		//添加边的关系
		void AddEdge(const V& src, const V& dst, const W& w)
		{
			size_t srci = GetVertexIndex(src);
			size_t dsti = GetVertexIndex(dst);
			_AddEdge(srci, dsti, w);
		}

	private:
		vector<V> _vertexs;               //顶点集合
		map<V, int> _vIndexMap;			  //顶点映射下标
		vector<vector<W>> _matrix;        //邻接矩阵
	};

关于图的构造,需要输入顶点,并且手动添加边。

添加边后,将邻接矩阵v->u的位置置为权值,如果是无向图,那么u->v也置上权值。


邻接矩阵的优缺点

  • 邻接矩阵的存储方式非常适合稠密图
  • 它能做到O(1)的效率判断俩个顶点的关系
  • 不适合找出所有邻接的边

邻接表

  • 邻接表:使用数组表示顶点的集合,使用链表表示边的关系
  • 邻接表是指针数组,将所有邻接的边,都挂在vector下。

一般而言,邻接表有入边和出边,我们只关心出边。

【数据结构】图的存储与遍历,数据结构,数据结构,算法,图论

边的结构 dsti w next

【数据结构】图的存储与遍历,数据结构,数据结构,算法,图论

对于邻接表就是将有关联的边都挂载在数组上。

同样需要vector数组保存顶点,map保存顶点和下标的映射。

邻接表的内容是一个边结构,内容包含srci(不一定有),dsti,w权值,next指向下一个关联的边。

namespace LinkTable
{
	template<class W>
	struct Edge
	{
		int _dsti;//目标点
		W _w; //权值
		Edge<W>* _next;
		Edge() = delete;

		Edge(int dsti,const W& w)
			:_dsti(dsti)
			,_w(w)
			,_next(nullptr)
		{
		}
	};
	template<class V, class W, bool Direction = false>
	class Graph
	{
		typedef Edge<W> Edge;
	public:
		Graph(const V* a, size_t n)
		{
			_vertexs.reserve(n);
			for (size_t i = 0; i < n; i++)
			{
				_vertexs.push_back(a[i]);
				_vIndexMap[a[i]] = i;
			}

			_tables.resize(n, nullptr);
		}

		//获取下标
		size_t GetVertexIndex(const V& v)
		{
			auto it = _vIndexMap.find(v);
			if (it != _vIndexMap.end())
			{
				return it->second;
			}
			else
			{
				throw invalid_argument("顶点不存在");
				assert(false);
				return -1;
			}
		}
		void _AddEdge(const size_t srci, const size_t dsti, const W& w)
		{
			//1->2
			Edge* eg = new Edge(dsti, w);
			eg->_next = _tables[srci];
			_tables[srci] = eg;
			
			//2->1
			if (Direction == false)
			{
				Edge* eg = new Edge(srci, w);
				eg->_next = _tables[dsti];
				_tables[dsti] = eg;
			}
		}

		void AddEdge(const V& src, const V& dst, const W& w)
		{
			size_t srci = GetVertexIndex(src);
			size_t dsti = GetVertexIndex(dst);
			_AddEdge(srci, dsti, w);
		}

		void Print()
		{
			//打印下标映射
			for (size_t i = 0; i < (_vertexs.size()); i++)
			{
				cout << "[" << i << "]" << "->" << _vertexs[i] << endl;
			}
			
			//打印边
			for (size_t i = 0; i < _tables.size(); i++)
			{
				cout << "[" << i << "]   ";
				Edge* cur = _tables[i];
				while (cur)
				{
					cout << "- [ dsti->" << cur->_dsti << "| w:"<<cur->_w<<"] -";
					cur = cur->_next;
				}
				cout << "null" << endl;
			}

		}
	private:
		vector<V> _vertexs;               //顶点集合
		map<V, int> _vIndexMap;			  //顶点映射下标
		vector<Edge*> _tables;         //邻接表(出边表)
	};

 邻接表的优缺点

  • 适合稠密图
  • 适合找顶点出去的边
  • 不适合用来确定俩个顶点的关系和权值

邻接矩阵和邻接表二者各有优势,相辅相成。在后续的最小生成树和最短路径中,邻接矩阵更方便


图的遍历

从v0出发,根据某种规则沿着图中各边访问图的顶点,每一个都会被访问到,且只被访问到一次。

遍历的方式分为广度优先BFS和深度优先DFS

广度优先BFS

广度优先类似于二叉树的层序遍历,从左到右依次遍历,一层到一层。

【数据结构】图的存储与遍历,数据结构,数据结构,算法,图论

BFS的思路

  • 要实现层序遍历,就要维护一个队列,A出队列时候,带A的相邻B C D入队列
  • 当 A 出完之后,B再出 就会带A进,为了防止重复带入,再维护一张vector的数组,出过了就标记,只有当标记容器没有被标记时,才带进队列。

【数据结构】图的存储与遍历,数据结构,数据结构,算法,图论

		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;
			
			while (!q.empty())
			{
				size_t front = q.front();
				q.pop();
				cout << front << ":" << _vertexs[front] << endl;
				for (size_t i = 0; i < _vertexs.size(); i++)
				{
					//入队列
					if (_matrix[front][i] != MAX_W)
					{
						if (visited[i] == false)
						{
							q.push(i);
							//修改标记
							visited[i] = true;
						}
					}
				}
			}
		}

【数据结构】图的存储与遍历,数据结构,数据结构,算法,图论


深度优先DFS

类似于二叉树的先序遍历,从上从往下,一直遍历到最深,知道遇到访问或结束,就返回。

【数据结构】图的存储与遍历,数据结构,数据结构,算法,图论

DFS需要一个起始点,标志从哪里开始遍历,需要一个visited数组,标记哪些顶点被访问过了

		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);
		}

【数据结构】图的存储与遍历,数据结构,数据结构,算法,图论

Gitee:提取完整代码文章来源地址https://www.toymoban.com/news/detail-836902.html

到了这里,关于【数据结构】图的存储与遍历的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

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

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

相关文章

  • 【数据结构与算法】图的搜索——广度优先遍历、最小生成树

    邻接链表: 用字典实现.有向图的邻接链表的总长度等于图的边个数;无向图的邻接链表的总长度等于图的边个数的2倍. 邻接矩阵:用于最短路径算法. 该数据结构维护一个不相交动态集的集合,每个集合有一个代表,不关心谁做代表。 支持三种操作: MAKE_SET(x) FIND_SET(x) UNION(x,y

    2024年02月19日
    浏览(36)
  • 算法数据结构——图的遍历之深度优先搜索算法(Depth First Search)

    深度优先搜索算法 (Depth First Search):英文缩写为 DFS。是一种用于搜索树或图的算法。所谓深度优先,就是说每次都尝试向更深的节点走。 深度优先搜索采用了回溯思想,该算法沿着树的深度遍历树的节点,会尽可能深的搜索树的分支。当节点 v 的所在边都己被探寻过,搜

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

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

    2024年02月05日
    浏览(39)
  • 【数据结构与算法】图的基本概念 | 邻接矩阵和邻接表 | 广度优先遍历和深度优先遍历

    🌠 作者:@ 阿亮joy. 🎆 专栏:《数据结构与算法要啸着学》 🎇 座右铭:每个优秀的人都有一段沉默的时光,那段时光是付出了很多努力却得不到结果的日子,我们把它叫做扎根 图是由顶点集合及顶点间的关系组成的一种数据结构:G = (V, E) ,其中: 顶点集合V = {x|x属于某

    2024年02月04日
    浏览(60)
  • 数据结构与算法基础-学习-24-图的遍历之DFS(深度优先搜索)和BFS(广度优先搜索)

    目录 一、遍历定义 二、遍历实质 三、DFS 四、BFS 五、宏定义 六、自定义类型 七、函数实现 1、DFS(邻接矩阵实现) 2、DFS(邻接表实现) 3、BFS(邻接矩阵实现) 4、BFS(邻接表实现) 5、打印邻接矩阵遍历顺序  6、打印邻接表遍历顺序 八、遍历算法效率分析 1、DFS 2、BFS 九

    2024年02月03日
    浏览(59)
  • 数据结构 | 图的遍历

    使用邻接矩阵法存储图的信息,其中 一维矩阵 Vexs[] 存储节点信息 二维矩阵 Edges[][] 存储边的信息 一维矩阵 visited[] 记录当前节点是否被访问过,用于后面的遍历 本文所使用的图的结构如下: 对应的 Vexs[] 为:A,B,C,D,E,F(对应的数组下标从0到5) 对应的 Edges[][] 如下: 图的

    2024年02月04日
    浏览(29)
  • 图论-图的基本概念与数据结构

    无向图 边是没有方向的,也就是双向的 结点 V = { v 1 , v 2 , . . . , v 7 } mathcal{V} = { v_1,v_2,...,v_7} V = { v 1 ​ , v 2 ​ , ... , v 7 ​ } 边 ε = { e 1 , 2 , e 1 , 3 , . . . , e 6 , 7 } varepsilon = {e_{1,2},e_{1,3},...,e_{6,7}} ε = { e 1 , 2 ​ , e 1 , 3 ​ , ... , e 6 , 7 ​ } 图 G = { V , ε } mathcal{G} = { math

    2024年02月08日
    浏览(38)
  • 数据结构--5.3图的遍历(广度优先遍历)

    广度优先遍历:         广度优先遍历(BreadthFirstSearch),又称为广度优先搜索,简称BFS。 要实现对图的广度遍历,我们可以利用队列来实现。  (参考队列)(上述为结构)

    2024年02月10日
    浏览(35)
  • 【数据结构】图的创建与遍历

    图(Graph)是由顶点的有穷非空集合和顶点之间边的集合组成,通常表示为:G(V,E),其中,G表示一个图,V是图G中顶点的集合,E是图G中边的集合。 线性表 :线性关系,由直接前驱和直接后继组成。 树 :层次关系,由父结点和孩子结点组成,每个结点最多有一个父结点(根

    2023年04月22日
    浏览(36)
  • 【数据结构】图的广度优先遍历

    广度优先遍历,类似于树的层次遍历,又是熟悉的队列实现。首先将第一个顶点添加到队列中,然后讲该顶点的所有邻接顶点都加入队列中,再将该顶点输出。如此重复直到遍历完整个图。 Q:队列,用于存放顶点。 front,rear:队头和队尾指针,用于入队和出队。 p:工作指针,用

    2024年02月05日
    浏览(36)

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

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

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

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

二维码1

领取红包

二维码2

领红包