图【数据结构】

news2025/1/11 6:08:23

目录

一、图的定义和基本术语

二、图的类型定义

三、图的存储结构

1、数组(邻接矩阵)表示法

二、邻接表(链式)表示法

三、图的邻接表的存储表示 

 四、十字链表与邻接多重链表

(1)十字链表

(2)邻接多重表

 五、图的遍历

(1)深度优先遍历(DFS):

(2)广度优先遍历 (BFS)

六、图的应用 

1、生成树 与 最小生成树

构造最小生成树 (MST性质)

构造最小生成树方法一: prim算法

构造最小生成树方法二:Kruskal

 2、最短路径

(1)单源最短路径------Dijkstra(迪杰斯特拉)算法O(n^3)

​编辑

(2)所有顶点间的最短路径------Floyd(费洛伊德)算法O(n^3)

3、拓扑排序 和 关键路径

(1)AOV网和拓扑排序  

 (2)AOE网和关键路径 


一、图的定义和基本术语

:G=(V,E)        Graph = (Vertex,Edge)

V:顶点(数据元素)的有穷非空集合

E:边的有穷集合

无向图:每条边都是无方向的。

有向图:每条边都是有方向的。

完全图:任意两个点都有一条边相连

稀疏图:有很少边或弧的图(e<nlogn)

稠密图:有较多边或弧的图

:边/弧带权的图

邻接:有边/弧相连的两个顶点之间的关系。

           存在(vi,vj)则称vi和vj互为邻接点;

           存在<vi,vj>则称vi邻接到vj,vj邻接于vi

关联(依附):边/弧与顶点之间的关系

        存在(vi,vj)/<vi,vj>,则称该边/弧关联于vi和vj

顶点的度:与该顶点相关联的边的数目,记TD(v)

在有向图中,顶点的读等于该顶点的出度与入度之和。

路径:接续的边构成的顶点序列。

路径长度:路径上边或弧的数目/权值之和。

回路(环):第一个顶点和最后一个顶点相同的路径。

简单路径:除路径起点和终点可以相同外,其余顶点均不相同的路径。

简单回路(简单环):除路径起点和终点相同外,其余顶点均不相同的路径。

连通图(强连通图):在无(有)向图G= (V,{E})中,若对任何两个顶点v、u都存在从v到u的路径,则称G是连通图(强连通图)。

权与网:图中边或弧所具有的相关数称为权,表明从一个顶点到另一个顶点的距离或耗费。

子图:a图边/弧、顶点都是b图的边/弧、顶点的子集,a图就是b图的子图。

连通分量(强连通分量)

无向图G的极大连通子图称为G的连通分量。

有向图G的极大强连通子图称为G的强连通分量。

极小连通子图:该子图是G的连通子图,在该子图中删除任何一条边/弧,子图不再连通。

生成树:包含无向图G所有顶点的极小连通子图。

生成森林:对非连通图,由各个连通分量的生成树的集合。

二、图的类型定义

图的抽象数据类型定义如下:
ADT Graph{
数据对象V:具有相同特性的数据元素的集合,称为顶点集。
数据关系R:R={VR}
VR={

        <V,W>|<v,W>|v,WEV^p(v,w),
       <v,w>表示从v到w的弧,P(v,w)定义了弧<v,w>的信息

}

基本操作P:
Create_Graph():图的创建操作。
初始条件:无。
操作结果:生成一个没有顶点的空图G。
GetVex(G,v):求图中的顶点v的值。
初始条件:图G存在,v是图中的一个顶点。
操作结果:生成一个没有顶点的空图G。

CreateGraph(&G,V,VR)
初始条件:V是图的顶点集,VR是图中弧的集合。
操作结果:按V和VR的定义构造图G。
DFSTraverse(G)
初始条件:图G存在。
操作结果:对图进行深度优先遍历。
BFSTraverse(G)
初始条件:图G存在。
操作结果:对图进行广度优先遍历。
}ADT Graph

三、图的存储结构

图的逻辑结构:多对多

图没有顺序存储结构,但可以借助二维数组来表示元素间的关系

重点介绍:

邻接矩阵(数组)表示法

邻接表(链式)表示法

1、数组(邻接矩阵)表示法

建立一个顶点表(记录各个顶点信息)和一个邻接矩阵(表示各个顶点之间关系)

 邻接矩阵:

(一)无向图的邻接矩阵表示法:

(1)无向图的邻接矩阵是对称的。

(2)顶点i的度 = 第i行(列)中1的个数;

(3)完全图的邻接矩阵中,对角线元素为0,其余为1。

(4)初始化邻接矩阵时,w均为0,构造邻接矩阵时,w为1

(二)有向图的邻接矩阵表示法

 注:行表示出度边,列表示入度边。

分析:

(1)有向图的邻接矩阵可能是不对称的。

(2)顶点的出度=第i行元素之和

        顶点的入度=第i列元素之和

顶点的度=第i行元素之和+第i列元素之和

(三)网(及有权图)的邻接矩阵表示法

1、有向网

2、无向网

(1)输入总项点数和总边数。

(2)依次输入点的信息存入顶点表中。

(3)初始化邻接矩阵,使每个取值初始化为极大值。

(4)构造邻接矩阵。

邻接矩阵-有什么好处?
(1)直观、简单、好理解
(2)方便检查任意一对顶点间是否存在边
(3)方便找任一顶点的所有“邻接点”(有边直接相连的顶点)
(4)方便计算任一顶点的“度”(从该点发出的边数为“出度”,指向该点的边数为“入度”)
(5)无向图:对应行(或列)非0元素的个数
(6)有向图:对应行非0元素的个数是“出度”;对应列非0元素的个数是“入度”

邻接矩阵-有什么坏处?
(1)不便于增加和删除顶点。
(2)浪费空间-存稀疏图(点很多而边很少)有大量无效元素。O(n^2)
(3)对稠密图(特别是完全图)还是很合算的。
(4)浪费时间-统计稀疏图中一共有多少条边。

二、邻接表(链式)表示法

顶点:按编号顺序将顶点数据存储在一维数组中;

关联同一顶点的边(以顶点为尾的弧):用线性链表存储。 

 (1)无向图:

特点:

(1)邻接表不唯一

(2)若无向图中有n个顶点,e条边,则其邻接表需n个头结点和2e个表结点。适合存储稀疏图。 空间:O(n+2e)

(3)无向图中vi的度为第i个单链表中结点数。

(2)有向图 

邻接表特点:

顶点vi的出度为第i个单链表中的结点个数。

顶点vi的入度为整个单链表中邻接点域值是i-1的结点个数。

找出度容易,找入度难。

相反的,逆邻接表找入度容易,找出度难。

三、图的邻接表的存储表示 

(1)顶点的结点结构

typedef struct VNode
{
	VerTexType data;//顶点信息
	ArcNode* firstarc;//指向第一条依附该顶点的边的指针
}VNode,AdjList[MVNum];//AdjList表示邻接表类型

(2)弧(边)的结点结构

typedef struct ArcNode
{
	int adjvex;
	struct ArcNode* nextarc;//指向下一条边
	OtherInfo info;//和边相关的信息
}ArcNode;

(3)图的结构定义

typedef struct
{
	AdjList vertices;//vertex的复数
	int vexnum, arcnum;//图的当前顶点数和弧数
}ALGraph;

采用邻接表法创建无向网

(1)输入总顶点数总边数

(2)建立顶点表

        依次输入点的信息存入顶点表中

        使每个表头结点的指针域初始化为NULL

(3)创建邻接表

        依次输入每条边依附的结点   

        确定两个顶点的序号i和j,建立边结点

        将此边结点分别插入到vi和vj对应的两个边链表的头部。

Status CreateUDG(ALGraph& G)
{
	cin >> G.vexnum >> G.arcnum;//输入总顶点数,总边数
	for (i = 0; i < G.vexnum; ++i)
	{
		cin >> G.vertices[i].data;//输入顶点值
		G.vertices[i].firstarc = NULL;//初始化表头结点的指针域
	}
	for (k = 0; k < G.vexnum; ++k)//输入各边,构造邻接表
	{
		cin >> v1 >> v2;//输入一条边依附的两个顶点
		j = LocateVex(G, v1);//找到两个顶点的下标
		i = LocateVex(G, v2);
	}
	p1 = new ArcNode;//生成一个新的边结点
	p1->adjvex = j;//邻接点序号为j
	p1->nextarc = G.vertices[i].firstarc;
	G.vertices[i].firstarc = p1;//将新节点*p1插入顶点vi的边表头部

	p2 = new ArcNode;//生成另一个对称的新的边结点*p2
	p2->adjvex = i;//邻接点序号为i
	p2->nextarc = G.vertices[i].firstarc;
	G.vertices[i].firstarc = p2;//将新节点*p2插入顶点vj的边表头部

	return ok;
}

 四、十字链表与邻接多重链表

(1)十字链表

十字链表(Orthogonal List)是有向图的另一种链式存储结构。我们也可以把它看成是将有向图的邻接表和逆邻接表结合起来形成的一种链表。
有向图中的每一条弧对应十字链表中的一个弧结点,同时有向图中的每个顶点在十字链表中对应有一个结点,叫做顶点结点。

(2)邻接多重表

边结点:

markivexilinkjvexjlinkinfo

mark:标记域,标记此边是否被搜索过。

ivex:该边依附的两个顶点在表头数组中位置

ilink:指向依附于ivex的下一条边

jvex:指向依附于jvex的下一条边

info:其他信息 

 五、图的遍历

(1)深度优先遍历(DFS):

方法:
■在访问图中某一起始顶点v后,由v出发,访问它的任一邻接顶点W1
■再从w1出发,访问与W邻接但还未被访问过的顶点W2
然后再从W2出发,进行类似的访问,...
如此进行下去,直至到达所有的邻接顶点都被访问过的顶点u为止。
接着,退回一步,退到前一次刚访问过的顶点,看是否还有其它没有被访问的邻接顶点。
如果有,则访问此顶点,之后再从此顶点出发,进行与前述类似的访问;
如果没有,就再退回一步进行搜索。重复上述过程,直到连通图中所有顶点都被访问过为止。

 

 

DFS算法效率分析:

用邻接矩阵来表示图,遍历图中每个顶点都要从头扫描该顶点所在行,时间复杂度为O(n^2)

用邻接表来表示图,虽然有2e个表结点,但只需要扫描e个结点即可完成遍历,加上访问n个头结点的时间,时间复杂度为O(n+e)。

稠密图适合在邻接矩阵上进行深度遍历。

稀疏图适合在邻接表上进行深度遍历。

(2)广度优先遍历 (BFS)

方法:从图的某一结点出发,首先依次访问该结点的所有邻点Vii, Vi2,..., Vin再按这些顶点被访问的先后次序依次访问与它们相邻接的所有未被访问的顶点
重复此过程,直至所有顶点均被访问为止。

广度优先遍历结果:v1->v2->v3->v4->v5->v6->v7->v8

BFS算法效率分析:

时间复杂度:

邻接矩阵:O(n^2)

邻接表:O(n+e)

空间复杂度:

都是O(n)(借用了堆栈或队列)

六、图的应用 

1、生成树 与 最小生成树

构造最小生成树 (MST性质)

 最小生成树:权值和最小的生成树,也叫最小代价生成树。

构造最小生成树方法一: prim算法

构造最小生成树方法二:Kruskal

 

 2、最短路径

(1)单源最短路径------Dijkstra(迪杰斯特拉)算法O(n^3)

(2)所有顶点间的最短路径------Floyd(费洛伊德)算法O(n^3)

 

  

3、拓扑排序 和 关键路径

AOV网解决拓扑排序问题

AOE网解决关键路径问题

(1)AOV网和拓扑排序  

 拓扑排序:将AOV网写成线性序列

拓扑序列并不唯一。

如何检测AOV网中是否存在环?

 如果网中所有顶点都在拓扑序列中,则不存在环。

 (2)AOE网和关键路径 

 关键路径

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

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

相关文章

113.删除有序数组中的重复项 removeDuplicatesFromSortedArray

文章目录 题目描述解题思路代码详解运行截图 题目描述 题目链接 给你一个 升序排列 的数组 nums &#xff0c;请你 原地 删除重复出现的元素&#xff0c;使每个元素 只出现一次 &#xff0c;返回删除后数组的新长度。元素的 相对顺序 应该保持 一致 。然后返回 nums 中唯一元…

Java中ReentrantLock的概念深入理解

ReentrantLock和Synchronized的区别 核心区别 ReentrantLock是一个类&#xff0c;Synchronized是Java中的一个关键字。 两者都是JVM层面实现互斥锁的方式 效率区别 线程竞争激烈推荐使用ReentrantLock去实现&#xff0c;不存在锁竞争观念&#xff1b; Synchronized是存在锁升…

大数据Doris(二十九):Broker Load导入HDFS csv 格式数据并提取文件路径中的分区字段

文章目录 Broker Load导入HDFS csv 格式数据并提取文件路径中的分区字段 一、创建Doris表 二、准备HDFS数据<

蚁群算法(解决TSP问题)

一、概述 蚂蚁在寻找食物源时&#xff0c;会在其经过的路径上释放一种信息素&#xff0c;并能够感知其它蚂蚁释放的信息素。信息素浓度的大小表征到食物源路径的远近&#xff0c;信息素浓度越高&#xff0c;表示对应的路径距离越短。通常&#xff0c;蚂蚁会以较大的概率优先…

chatgpt赋能python:Python文件复制粘贴到另一个目录

Python文件复制粘贴到另一个目录 Python是一种通用编程语言&#xff0c;可用于各种任务&#xff0c;包括文件复制和移动。在本文中&#xff0c;我们将探讨Python中的文件复制粘贴到另一个目录。 为什么要使用Python进行文件复制粘贴&#xff1f; Python提供了强大的文件处理…

case when用法

case when的基本使用&#xff1a; Case when 的用法: 一旦满足了某一个WHEN, 则这一条数据就会退出CASE WHEN , 而不再考虑 其他CASE。 Case when 的用法 -- -搜索Case函数: Case函数(Case搜索函数): 判断表达式的真假,如果为真,返回结果;如果为假,返回else值;如果未定义el…

批处理文件(.bat)启动redis及任何软件(同理)

批处理文件 每次从文件根目录用配置文件格式来启动redis太麻烦了 可以在桌面上使用批处理文件&#xff08;.bat&#xff09;启动Redis&#xff0c;请按照以下步骤进行操作&#xff1a; 打开文本编辑器&#xff0c;如记事本。 在编辑器中输入以下内容&#xff1a; 将文件保存…

70.爬楼梯问题+746.使用最小花费爬楼梯

目录 一、70.爬楼梯问题分析 二、代码 三、746.使用最小花费爬楼梯分析 四、代码 一、70.爬楼梯问题分析 70. 爬楼梯 - 力扣&#xff08;LeetCode&#xff09; 二、代码 class Solution { public:int climbStairs(int n) {if(n1||n2)return n;vector<int>dp(n1);dp…

线性表的链式表示——单链表

目录 一、单链表的定义二、单链表上基本操作的实现1、采用头插法建立单链表2、采用尾插法建立单链表3、按序号查找结点值4、按值查找表结点5、插入结点操作6、删除结点操作7、求表长操作 三、双链表、循环链表、静态链表 顺序表可以随时存取表中的任意一个元素&#xff0c;它的…

翻译:开源软件的架构(volume2): 可伸缩web框架及分布式系统

英文源地址 开源软件已经成为构建一些超大型网站的基础组成部分了.随着这些网站的成长, 围绕着它们软件架构的最佳实践与指导思想开始涌现.本文尝试去阐述设计大型网站时的需要考虑一些关键问题, 以及用于实现这些目标的基础组件. 本文主要关注web系统,尽管其中一些内容也适用于…

网络安全-01-VMware安装Kali部署DVWA

网络安全-01-VMware安装Kali&部署DVWA &#x1f53b;一、Kali简介&下载&#x1f4d7; 二、VMware安装Kali&#x1f4f0; 2.1 新建虚拟机&#x1f4f0; 2.2 开始安装Kali&#x1f4f0; 2.3 更换apt源为国内源&#x1f4f0; 2.4 启动mysql-这里使用自带的maridb&#x1f…

【Python 垃圾回收】零基础也能轻松掌握的学习路线与参考资料

Python 垃圾回收是 Python 运行机制中的重要环节。了解 Python 垃圾回收机制可帮助开发者高效编写 Python 代码&#xff0c;并避免潜在的内存泄漏问题。本文将介绍 Python 垃圾回收的学习路线&#xff0c;并给出参考资料和优秀实践。 Python 垃圾回收机制 Python 使用引用计数…

关于社会脑研究的fMRI和fNIRS超扫描方法

导读 近来&#xff0c;“社会脑”(即大脑在社会情境中是如何工作的&#xff0c;以及我们社会行为的机制是什么)在神经科学文献中获得了很多关注&#xff0c;主要是因为最近开发的技术允许研究人类社会认知的不同方面及其与大脑的关联。在这种情况下&#xff0c;超扫描技术拓宽…

如何在华为OD机试中获得满分?Java实现【质数因子】一文详解!

✅创作者:陈书予 🎉个人主页:陈书予的个人主页 🍁陈书予的个人社区,欢迎你的加入: 陈书予的社区 🌟专栏地址: Java华为OD机试真题(2022&2023) 文章目录 1. 题目描述2. 输入描述3. 输出描述4. Java算法源码5. 测试6.解题思路1. 题目描述 功能:输入一个正整数,…

vue小案例TodoList

1.首先我们分析怎么把一个页面拆成多个组件&#xff0c;如下图&#xff0c;我们可以拆成MyHeader、MyList、MyItem、MyFooter&#xff0c;其中MyList包含MyItem 2.观看如下代码&#xff08;我们把MyItem作为MyList的子组件&#xff0c;在父组件中使用v-for指令来循环展示子组件…

【java基础】Map集合

大家好&#x1f44b;&#xff0c;今天我来给大家科普一下Java中的map集合。map是Java中非常重要的数据结构之一&#xff0c;经常被用于存储键值对。 【有关这部分知识的思维导图放在文章末尾了&#xff0c;需要的C友请自取】 正文开始&#xff1a; 一、Map集合概述 我们知道&…

柱状图中最大的矩形

题目链接 柱状图中最大的矩形 题目描述 注意点 无 解答思路 暴力破解根据每根柱子x以x的高度作为矩形的高度找到其相邻能组成矩形的柱子&#xff0c;遍历所有柱子即可找到最大矩形&#xff0c;但是时间复杂度是O(n)&#xff0c;最终运行结果也超时了上面暴力破解的方法中…

MKS SERVO4257D 闭环步进电机_系列4 MODBUS指令说明

第1部分 产品介绍 MKS SERVO 28D/35D/42D/57D 系列闭环步进电机是创客基地为满足市场需求而自主研发的一款产品。具备脉冲接口和RS485/CAN串行接口&#xff0c;支持MODBUS-RTU通讯协议&#xff0c;内置高效FOC矢量算法&#xff0c;采用高精度编码器&#xff0c;通过位置反馈&am…

基于Android的视频分享平台的设计与实现

基于Android的视频分享平台的设计与实现 摘 要 短视频平台是以特定群众为目标的差异化群体定位工具。其利用自身的便捷性可以实现视频的随时拍摄和随时上传&#xff0c;可以产生亚文化圈的萌芽。这种开放便利的特性在吸引了广大用户的同时&#xff0c;也在一定程度上解决了由…

C#面试题 2

七、AOP如何在代码层面实现&#xff1f; ①静态代理实现AOP&#xff0c;自己写代理对象&#xff1b; ②动态代理实现AOP&#xff0c;运行时动态生成一个代理对象。 静态代理的实现&#xff1a;①装饰器模式&#xff08;允许向一个现有的对象添加新的功能&#xff0c;同时又不…