【图与网络数学模型】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

相关文章

解决pandas无法读取csv文件数据的问题

《解决pandas无法读取csv文件数据的问题》本文讲述作者用Pandas读取CSV文件时因参数设置不当导致数据错位,通过调整delimiter和on_bad_lines参数最终解决问题,并强调正确参... 目录一、前言二、问题复现1. 问题2. 通过 on_bad_lines=‘warn’ 跳过异常数据3

解决RocketMQ的幂等性问题

《解决RocketMQ的幂等性问题》重复消费因调用链路长、消息发送超时或消费者故障导致,通过生产者消息查询、Redis缓存及消费者唯一主键可以确保幂等性,避免重复处理,本文主要介绍了解决RocketM... 目录造成重复消费的原因解决方法生产者端消费者端代码实现造成重复消费的原因当系统的调用链路比较长的时

python设置环境变量路径实现过程

《python设置环境变量路径实现过程》本文介绍设置Python路径的多种方法:临时设置(Windows用`set`,Linux/macOS用`export`)、永久设置(系统属性或shell配置文件... 目录设置python路径的方法临时设置环境变量(适用于当前会话)永久设置环境变量(Windows系统

深度解析Nginx日志分析与499状态码问题解决

《深度解析Nginx日志分析与499状态码问题解决》在Web服务器运维和性能优化过程中,Nginx日志是排查问题的重要依据,本文将围绕Nginx日志分析、499状态码的成因、排查方法及解决方案展开讨论... 目录前言1. Nginx日志基础1.1 Nginx日志存放位置1.2 Nginx日志格式2. 499

kkFileView启动报错:报错2003端口占用的问题及解决

《kkFileView启动报错:报错2003端口占用的问题及解决》kkFileView启动报错因office组件2003端口未关闭,解决:查杀占用端口的进程,终止Java进程,使用shutdown.s... 目录原因解决总结kkFileViewjavascript启动报错启动office组件失败,请检查of

SpringBoot 异常处理/自定义格式校验的问题实例详解

《SpringBoot异常处理/自定义格式校验的问题实例详解》文章探讨SpringBoot中自定义注解校验问题,区分参数级与类级约束触发的异常类型,建议通过@RestControllerAdvice... 目录1. 问题简要描述2. 异常触发1) 参数级别约束2) 类级别约束3. 异常处理1) 字段级别约束

Python错误AttributeError: 'NoneType' object has no attribute问题的彻底解决方法

《Python错误AttributeError:NoneTypeobjecthasnoattribute问题的彻底解决方法》在Python项目开发和调试过程中,经常会碰到这样一个异常信息... 目录问题背景与概述错误解读:AttributeError: 'NoneType' object has no at

Spring的RedisTemplate的json反序列泛型丢失问题解决

《Spring的RedisTemplate的json反序列泛型丢失问题解决》本文主要介绍了SpringRedisTemplate中使用JSON序列化时泛型信息丢失的问题及其提出三种解决方案,可以根据性... 目录背景解决方案方案一方案二方案三总结背景在使用RedisTemplate操作redis时我们针对

Kotlin Map映射转换问题小结

《KotlinMap映射转换问题小结》文章介绍了Kotlin集合转换的多种方法,包括map(一对一转换)、mapIndexed(带索引)、mapNotNull(过滤null)、mapKeys/map... 目录Kotlin 集合转换:map、mapIndexed、mapNotNull、mapKeys、map

nginx中端口无权限的问题解决

《nginx中端口无权限的问题解决》当Nginx日志报错bind()to80failed(13:Permissiondenied)时,这通常是由于权限不足导致Nginx无法绑定到80端口,下面就来... 目录一、问题原因分析二、解决方案1. 以 root 权限运行 Nginx(不推荐)2. 为 Nginx