【图与网络数学模型】1.Dijkstra算法求解最短路径问题

2024-02-19 08:20

本文主要是介绍【图与网络数学模型】1.Dijkstra算法求解最短路径问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述

【图与网络数学模型】1.Dijkstra算法求解最短路径问题

  • 一、图论基本概念
    • 1. 图论
    • 2. 哥尼斯堡七桥问題
    • 3. 图的一些基本概念
    • 4. 图的矩阵表示
  • 二、最短路径问题算法
    • 1. 图形的标号法
    • 2. Dijkstra法
      • (1)基本思路
      • (2)求解示例
  • 三、基于Python的Dijkstra算法
    • 1. 创建图对象
    • 2. 打印最短路径
    • 3. 查找顶点
    • 4. Dijkstra 算法实现
    • 5. 完整程序运行


一、图论基本概念

1. 图论

图论是应用非常广泛的运筹学分支,它已经广泛地应用于物理学控制论、信息论、工程技术、交通运输、经济管理、电子计算机等各项领域。对于科学研究、市场和社会生活中的许多问题,可以用图论的理论和方法来加以解决。

2. 哥尼斯堡七桥问題

关于图的第一篇论文是瑞士数学家欧拉(E. Euler)在1736年发表的解决“哥尼斯堡” 七桥难题的论文。

要如何走过每座桥恰一次,再返回出发点?

在这里插入图片描述

即能否从某一点开始不重复地一笔画出这个图形,最终回到原点。欧拉在他的论文中证明了这是不可能的,因为这个图形中每一个顶点都与奇数条边相连接,不可能将它一笔画出,这就是古典图论中的第一个著名问题。

3. 图的一些基本概念

图论中的图是由点、点与点之间的线所组成的。

涉及的一些概念如下表,在此不做赘述。

边,弧无向图
端点关联边有向图
始点,终点多重边简单图初等链/圈
度(次)多重图简单链/圈
奇点,偶点连通图基础图
悬挂点悬挂边不连通图
弧立点回路

4. 图的矩阵表示

邻接矩阵 Adjacency matrix

  • 表示图中两点之间的相互关系
  • 两点之间有弧或边的,用1表示,否则用0表示,构成一个矩阵A

在这里插入图片描述
上图可用矩阵表示如下:
在这里插入图片描述

二、最短路径问题算法

最短路径问题是图论中十分重要的最优化问题之一,它作为一个经常被用到的基本工具,可以解决生产实际中的许多问题,也可以用于解决其它的最优化问题。

如果P是D中从 v s v_s vs v i v_i vi 的最短路, v i v_i vi 是 P 中的一个点,那么从 v s v_s vs 沿 P 到 v i v_i vi 的路是从 v s v_s vs v i v_i vi 的最短路。

1. 图形的标号法

求下图从A到G的最短路:
在这里插入图片描述

解法:先标出离终点最近的一段,然后标号与该点距离最短的点,继续逆推至始点。

在这里插入图片描述

从A到G的最短路为: A - B1-C2-D1-E2-F2-G

2. Dijkstra法

(1)基本思路

v s v_s vs出发,逐步地向外探寻最短路。

执行过程中,与每个点对应,记录下一个数 (称为此点的标号)

  1. 它或者表示从 v s v_s vs到该点的最短路的权 (称为P标号)
  2. 或者是从 v s v_s vs到该点的最短路的权的上界 (称为T标号)

方法的每一步是修改T标号,并且把某一个具T标号的点改变为具P标号的点,从而使D中具P标号的点多一个,如此经过 p-1步,就可以求出从 v s v_s vs到各点的最短路。

(2)求解示例

例如求下图从1到8的最短路径:
在这里插入图片描述

  1. X={1}, w 1 w_1 w1=0
    在这里插入图片描述

  2. X={1,4}
    在这里插入图片描述

  3. X={1,2,4}
    在这里插入图片描述

  4. X={1,2,4,6}
    在这里插入图片描述

  5. X={1,2,4,6,7}
    在这里插入图片描述

  6. X={1,2,3,4,6,7}
    在这里插入图片描述

  7. X={1,2,3,4,6,7,8}
    在这里插入图片描述
    1到8的最短路径为{1,4,7,5,8},长度为10。

三、基于Python的Dijkstra算法

1. 创建图对象

创建一个图对象:接受一个参数 vertices,表示图中顶点的数量。

在初始化过程中,将顶点数量存储在 self.V 中,并创建一个二维数组 self.graph 作为邻接矩阵,初始值为0。

	def __init__(self, vertices):self.V = verticesself.graph = [[0 for column in range(vertices)]for row in range(vertices)]

2. 打印最短路径

printSolution(self, dist):接受一个参数 dist,该参数是一个包含各个顶点距离源点的最短距离的列表。

遍历所有顶点,打印每个顶点及其距离源点的最短距离。

	def printSolution(self, dist):print("Vertex \t Distance from Source")for node in range(self.V):print(node, "\t\t", dist[node])

3. 查找顶点

查找未包含在最短路径树中的距离源点最近的顶点。

minDistance(self, dist, sptSet):接受两个参数 dist 和 sptSet,分别表示从源点到各个顶点的当前最短距离和是否已经包含在最短路径树中。

遍历所有顶点,找到距离源点最近且未包含在最短路径树中的顶点,并返回其索引。

	def minDistance(self, dist, sptSet):min = 1e7for v in range(self.V):if dist[v] < min and sptSet[v] == False:min = dist[v]min_index = vreturn min_index

4. Dijkstra 算法实现

参数 src:表示源点的索引。

初始化距离源点的距离列表 dist 和最短路径树标记列表 sptSet。

在循环中,选择距离源点最近且未包含在最短路径树中的顶点作为当前顶点 u,将其标记为已经包含在最短路径树中。

更新与当前顶点相邻且未包含在最短路径树中的顶点的距离,如果新的距离小于之前记录的距离,则更新距离值。

最后调用 printSolution 方法打印结果。

	def dijkstra(self, src):dist = [1e7] * self.Vdist[src] = 0sptSet = [False] * self.Vfor cout in range(self.V):u = self.minDistance(dist, sptSet)sptSet[u] = Truefor v in range(self.V):if (self.graph[u][v] > 0 andsptSet[v] == False anddist[v] > dist[u] + self.graph[u][v]):dist[v] = dist[u] + self.graph[u][v]self.printSolution(dist)

5. 完整程序运行

求下图从1到8的最短路径:
在这里插入图片描述

class Graph():def __init__(self, vertices):self.V = verticesself.graph = [[0 for column in range(vertices)]for row in range(vertices)]def printSolution(self, dist):print("Vertex \t Distance from Source")for node in range(self.V):print(node, "\t\t", dist[node])def minDistance(self, dist, sptSet):min = 1e7for v in range(self.V):if dist[v] < min and sptSet[v] == False:min = dist[v]min_index = vreturn min_indexdef dijkstra(self, src):dist = [1e7] * self.Vdist[src] = 0sptSet = [False] * self.Vfor cout in range(self.V):u = self.minDistance(dist, sptSet)sptSet[u] = Truefor v in range(self.V):if (self.graph[u][v] > 0 andsptSet[v] == False anddist[v] > dist[u] + self.graph[u][v]):dist[v] = dist[u] + self.graph[u][v]self.printSolution(dist)g = Graph(8)
g.graph = [[0, 2, 0, 1, 0, 3, 0, 0],[0, 0, 6, 0, 5, 0, 0, 0],[0, 0, 0, 0, 0, 0, 0, 6],[0, 10, 0, 0, 0, 0, 2, 0],[0, 0, 9, 0, 0, 0, 0, 4],[0, 0, 0, 5, 0, 0, 4, 0],[0, 7, 0, 0, 3, 0, 0, 8],[0, 0, 0, 0, 0, 0, 0, 0]]g.dijkstra(0)

输出结果如下:

Vertex 	 Distance from Source
0 		 0
1 		 2
2 		 8
3 		 1
4 		 6
5 		 3
6 		 3
7 		 10

1-8最短路径的长度为10.

这篇关于【图与网络数学模型】1.Dijkstra算法求解最短路径问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/724095

相关文章

IDEA和GIT关于文件中LF和CRLF问题及解决

《IDEA和GIT关于文件中LF和CRLF问题及解决》文章总结:因IDEA默认使用CRLF换行符导致Shell脚本在Linux运行报错,需在编辑器和Git中统一为LF,通过调整Git的core.aut... 目录问题描述问题思考解决过程总结问题描述项目软件安装shell脚本上git仓库管理,但拉取后,上l

深入理解Mysql OnlineDDL的算法

《深入理解MysqlOnlineDDL的算法》本文主要介绍了讲解MysqlOnlineDDL的算法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小... 目录一、Online DDL 是什么?二、Online DDL 的三种主要算法2.1COPY(复制法)

idea npm install很慢问题及解决(nodejs)

《ideanpminstall很慢问题及解决(nodejs)》npm安装速度慢可通过配置国内镜像源(如淘宝)、清理缓存及切换工具解决,建议设置全局镜像(npmconfigsetregistryht... 目录idea npm install很慢(nodejs)配置国内镜像源清理缓存总结idea npm in

pycharm跑python项目易出错的问题总结

《pycharm跑python项目易出错的问题总结》:本文主要介绍pycharm跑python项目易出错问题的相关资料,当你在PyCharm中运行Python程序时遇到报错,可以按照以下步骤进行排... 1. 一定不要在pycharm终端里面创建环境安装别人的项目子模块等,有可能出现的问题就是你不报错都安装

idea突然报错Malformed \uxxxx encoding问题及解决

《idea突然报错Malformeduxxxxencoding问题及解决》Maven项目在切换Git分支时报错,提示project元素为描述符根元素,解决方法:删除Maven仓库中的resolv... 目www.chinasem.cn录问题解决方式总结问题idea 上的 maven China编程项目突然报错,是

Python爬虫HTTPS使用requests,httpx,aiohttp实战中的证书异步等问题

《Python爬虫HTTPS使用requests,httpx,aiohttp实战中的证书异步等问题》在爬虫工程里,“HTTPS”是绕不开的话题,HTTPS为传输加密提供保护,同时也给爬虫带来证书校验、... 目录一、核心问题与优先级检查(先问三件事)二、基础示例:requests 与证书处理三、高并发选型:

前端导出Excel文件出现乱码或文件损坏问题的解决办法

《前端导出Excel文件出现乱码或文件损坏问题的解决办法》在现代网页应用程序中,前端有时需要与后端进行数据交互,包括下载文件,:本文主要介绍前端导出Excel文件出现乱码或文件损坏问题的解决办法,... 目录1. 检查后端返回的数据格式2. 前端正确处理二进制数据方案 1:直接下载(推荐)方案 2:手动构造

利用Python把路径转为绝对路径的方法

《利用Python把路径转为绝对路径的方法》在Python中,如果你有一个相对路径并且想将其转换为绝对路径,你可以使用Path对象的resolve()方法,Path是Python标准库pathlib中... 目录1. os.path.abspath 是什么?怎么用?基本用法2. os.path.abspat

Python绘制TSP、VRP问题求解结果图全过程

《Python绘制TSP、VRP问题求解结果图全过程》本文介绍用Python绘制TSP和VRP问题的静态与动态结果图,静态图展示路径,动态图通过matplotlib.animation模块实现动画效果... 目录一、静态图二、动态图总结【代码】python绘制TSP、VRP问题求解结果图(包含静态图与动态图

MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决

《MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决》MyBatis默认开启一级缓存,同一事务中循环调用查询方法时会重复使用缓存数据,导致获取的序列主键值均为1,... 目录问题原因解决办法如果是存储过程总结问题myBATis有如下代码获取序列作为主键IdMappe