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

news2025/3/10 21:11:28

图的概念

图是由顶点集合及顶点间的关系组成的一种数据结构: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:提取完整代码

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

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

相关文章

C++学习:list

1.list的定义和结构 list的使用频率不高&#xff0c;在做题时几乎遇不到需要使用list的情景。list是一种双向链表容器&#xff0c;它是标准模板库(STL)提供的一种序列容器。list容器以节点(node的形式存储元素&#xff0c;并使用指针将这些节点链接在一起&#xff0c;形成一个…

【快速解决】python项目打包成exe文件——vscode软件

目录 操作步骤 1、打开VSCode并打开你的Python项目。 2、在VSCode终端中安装pyinstaller&#xff1a; 3、运行以下命令使用pyinstaller将Python项目打包成exe文件&#xff1a; 其中your_script.py是你的Python脚本的文件名。 4、打包完成后&#xff0c;在你的项目目录中会…

如何通过软文引起用户共鸣,媒介盒子支招

不管是哪个行业&#xff0c;哪个品牌都需要通过软文来吸引用户&#xff0c;一篇合格的软文应该能引起用户情绪&#xff0c;让用户为情绪买单&#xff0c;引起用户的共鸣&#xff0c;今天媒介盒子就来和大家聊聊&#xff1a;如何通过软文引起用户共鸣。 一、 熟知用户心理情绪 …

开源软件:推动软件行业繁荣的力量

文章目录 &#x1f4d1;引言开源软件的优势分析开放性与透明度低成本与灵活性创新与协作 开源软件对软件行业的影响推动技术创新和进步促进软件行业的合作与交流培养人才和提高技能促进软件行业的可持续发展 结语 &#x1f4d1;引言 随着信息技术的飞速发展&#xff0c;软件已经…

力扣热题100_双指针_11_盛最多水的容器

文章目录 题目链接解题思路解题代码 题目链接 11. 盛最多水的容器 给定一个长度为 n 的整数数组 height 。有 n 条垂线&#xff0c;第 i 条线的两个端点是 (i, 0) 和 (i, height[i]) 。 找出其中的两条线&#xff0c;使得它们与 x 轴共同构成的容器可以容纳最多的水。 返回…

洛谷P5716 月份天数 题解

#题外话&#xff08;第31篇题解&#xff09;&#xff08;累了&#xff0c;歇会……&#xff09; #先看题目 题目链接https://www.luogu.com.cn/problem/P5716 #思路&#xff08;看代码&#xff09; #代码 代码1&#xff1a;不管三七二十七&#xff0c;直接先判断闰年&#x…

网络原理 - HTTP/HTTPS(1)

HTTP HTTP是什么 HTTP("全程超文本协议")是一种应用非常广泛的应用层协议. 文本:字符串(能在utf8/gbk)码表上找到合法字符. 超文本:不仅是字符串,还能携带图片啥的(HTML). 富文本:类似于word文档这种. HTTP诞生于1991年.目前已经发展为最主流使用的一种应用层协议.…

[Docker实战] 旭日X3派上Docker Openwrt +Samba 实现局域网NAS 开启AP模式

​ &#x1f308; 博客个人主页&#xff1a;Chris在Coding &#x1f3a5; 本文所属专栏&#xff1a;[旭日X3派] [Docker实战] ❤️ 前置学习专栏&#xff1a;[Linux学习] ⏰ 我们仍在旅途 …

Python:变量与数据类型

目录 一、变量 1.1 强数据类型与弱数据类型 1.2 全局函数 1.3 变量的命名规范 二、数据类型 2.1 基本数据类型 2.2 复合数据类型&#xff08;引用数据类型&#xff09; 三、数据类型转换 一、变量 变量&#xff1a;顾名思义&#xff0c;变化的量。在python中代指运行时…

博客新增每日早报api,网站增加每日早报功能

1、每日早报 使用alapi的每日早报接口&#xff0c;回调一个日报图片&#xff0c;然后展示这个图片&#xff0c;即可看到每日早报内容 1.1 api申请 在Alapi官网注册一个账号 1.2 获取密钥 然后获取用户中心的Token密钥&#xff0c;这个用于输出早报内容 2、早报创建 2.1 创…

2024最新软件测试八股文(答案+文档)

&#x1f345; 视频学习&#xff1a;文末有免费的配套视频可观看 &#x1f345; 关注公众号【互联网杂货铺】&#xff0c;回复 1 &#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 1、B/S架构和C/S架构区别 B/S 只需要有操作系统和浏览器就行&a…

自然语言编程系列(一):自然语言和程序语言介绍

1.自然语言和程序语言 自然语言和程序语言是两种截然不同但又相互关联的语言体系&#xff0c;它们分别服务于人类日常交流和计算机指令执行。 自然语言&#xff1a; 定义&#xff1a;自然语言是指人类在日常生活中使用的语言&#xff0c;如英语、汉语、法语等。它是非正式且灵…

2024年回炉计划之JWT(五)

一、简介 WT&#xff08;JSON Web Token&#xff09;是一种用于在网络应用间安全地传递信息的开放标准&#xff08;RFC 7519&#xff09;。它是一种紧凑且自包含的方式&#xff0c;用于在各方之间传输信息作为 JSON 对象。JWT 可以通过数字签名&#xff08;使用 HMAC 算…

Kafka(二)

第 4 章 Kafka Broker 4.1 Kafka Broker 工作流程 4.1.1 Zookeeper 存储的 Kafka 信息 &#xff08;1&#xff09;启动 Zookeeper 客户端。 bin/zkCli.sh &#xff08;2&#xff09;通过 ls 命令可以查看 kafka 相关信息。 ls /kafka 4.1.2 Kafka Broker 总体工作流程…

C++类和对象——继承详解

目录 1.基本语法 2.继承方式 3.继承中的对象模型 4.构造和构析顺序 5.同名成员处理 6.同名静态成员处理 7.多继承语法 8.菱形继承 图片示例&#xff1a; 虚继承 代码示例&#xff1a; 1.基本语法 #include<bits/stdc.h> using namespace std;//公共页面类 …

招募Sui大使,共同构建Sui社区,解锁专属福利

我们非常激动地宣布推出新一轮的Sui大使计划&#xff0c;这是围绕Sui创新技术构建全球社区的关键举措。 大使计划赋予了热衷于Sui使命并渴望在Sui社区和受众中传播意识的个人以权力。Sui大使体现了网络的价值观&#xff0c;并通过战略性和有影响力的行动加速了采用过程。我们很…

PFA洗气瓶配空气采样泵用PFA气体吸收瓶的特点

PFA洗气瓶是一种洗去气体中杂质的器皿&#xff0c;是将不纯气体通过选定的适宜液体介质鼓泡吸收&#xff08;溶解或由于发生化学反应&#xff09;&#xff0c;从而洗去杂质气体&#xff0c;以达净化气体的目的。在设计时&#xff0c;四氟球的周围都布满小孔。一般情况下&#x…

在字节划水的7年,太真实了。。

&#x1f345; 视频学习&#xff1a;文末有免费的配套视频可观看 &#x1f345; 关注公众号【互联网杂货铺】&#xff0c;回复 1 &#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 测试这条路是坎坷的&#xff0c;我自己深有体会。 我们的起点低…

02_debugfsLinux内核模块

01_basicLinux内核模块-CSDN博客环境IDubuntuMakefilemodules:clean:basic.creturn 0;运行效果。https://blog.csdn.net/m0_37132481/article/details/136157384?csdn_share_tail%7B%22type%22%3A%22blog%22%2C%22rType%22%3A%22article%22%2C%22rId%22%3A%22136157384%22%2C%…

分享一个学英语的网站

名字叫&#xff1a;公益大米网​​​​​​​ Freerice 这个网站是以做题的形式来记忆单词&#xff0c;题干是一个单词&#xff0c;给出4个选项&#xff0c;需要选出其中最接近题干单词的选项。 答对可以获得10粒大米&#xff0c;网站的创办者负责捐赠。如图 触发某些条件&a…