最短路径之Dijkstra算法和Floyd-Warshall算法
最短路径算法
最短路径算法通常用在寻找图中任意两个结点之间的最短路径或者是求全局最短路径,像是包括Dijkstra、A*、Bellman-Ford、SPFA(Bellman-Ford的改进版本)、Floyd-Warshall、Johnson、BFS等等,这里要集中介绍Dijkstra和Floyd,前者用来处理任意两个结点之间的最短路径,后者处理图中所有的最短路径。
Dijkstra算法
理解
其思想为,我们首先纪录下每个点到原点的距离,这个距离会在每一轮遍历的过程中刷新。每一个结点到原点的最短路径是其前驱结点到原点的最短路径加上前驱结点到当前结点的路径和。
我们以下面这幅图片为例,假定起始结点为1,求该结点到其余各个结点的最短路径。

执行步骤为:
| 步骤 | 描述 | 前驱结点 |
|---|---|---|
| 1 | 将当前结点到其它结点的路径都初始化为无穷大 | 无 |
| 2 | 依次计算每个结点的路径长度,此时可以获得结点1到其它的结点2、3、4、5、6的路径长度为[7, 9, ∞,∞,14]`,取其中最小的值7,结点2作为前驱结点 | 无 |
| 3 | 此时通过结点2可以到达结点4和5,并且其路径等于起始结点到前驱结点的路径加上前驱结点到目标结点的路径长度,因此此时结点1到其它结点3、4、5、6的路径长度为[17, 22, ∞, ∞],此时将现在的路径长度和之前的路径长度进行比较,取对应的较小值,得到最短距离[9, 22, ∞, 14],取其最小值9,结点3更新成前驱结点 | 2 |
| 4 | 同样的道理,此时结点1到其它的结点4、5、6的路径长度为[20, ∞, 11],和之前的路径长度进行比较,得到最短距离[20, ∞, 11],取最小值11,结点6作为前驱结点 | 3 |
| 5 | 同样的道理,得到结点1到其它的结点4, 5的路径长度为[20, 20],和之前对比,最短距离更新成[20,20] | 6 |
| 6 | 因为此时剩下了两个相等路径长度的结点,取任意一个结点,结果都是一样的,所以取哪个都可以 |
其伪函数如下:
function Dijkstra(G, w, s)
for each vertex v in V[G] # 初始化
d[v] := infinity # 將各點的已知最短距離先設成無窮大
previous[v] := undefined # 各点的已知最短路径上的前趋都未知
d[s] := 0 # 因为出发点到出发点间不需移动任何距离,所以可以直接将s到s的最小距离设为0
S := empty set
Q := set of all vertices
while Q is not an empty set # Dijkstra演算法主體
u := Extract_Min(Q)
S.append(u)
for each edge outgoing from u as (u,v)
if d[v] > d[u] + w(u,v) # 拓展边(u,v)。w(u,v)为从u到v的路径长度。
d[v] := d[u] + w(u,v) # 更新路径长度到更小的那个和值。
previous[v] := u # 紀錄前趨頂點
其中的Extract_Min为提取最小值。
实现
import sys
class Graph():
def __init__(self, vertices):
self.V = vertices
self.graph = [[0 for _ in range(self.V)] for _ in range(self.V)]
# 打印距离
def print_distance(self, dist, parent):
for node in range(self.V):
print('distance')
print(node, dist[node])
print('path')
self.print_path(parent, node)
print('---------')
# 打印路径
def print_path(self, parent, j):
if parent[j] == -1:
print(j)
return
self.print_path(parent, parent[j])
print(j)
# 找出最小距离的结点
def min_distance(self, dist, sptSet):
min_val = sys.maxsize
min_index = 0
for v in range(self.V):
if dist[v] < min_val and not sptSet[v]:
min_val = dist[v]
min_index = v
return min_index
# 核心算法
def dijkstra(self, src):
dist = [sys.maxsize] * self.V
dist[src] = 0
sptSet = [False] * self.V
parent = [-1] * self.V
for count in range(self.V):
# 找出前驱结点
u = self.min_distance(dist, sptSet)
sptSet[u] = True
# 如果某个节点的路径大于经过前驱结点的路径,则更新结果成经过前驱结点的路径
for v in range(self.V):
if self.graph[u][v] > 0 and not sptSet[v] and dist[v] > dist[u] + self.graph[u][v]:
dist[v] = dist[u] + self.graph[u][v]
parent[v] = u
self.print_distance(dist, parent)
g = Graph(8)
g.graph = [[0, 0, 7, 7, 2, 0, 0, 0],
[0, 0, 3, 8, 0, 0, 8, 0],
[7, 3, 0, 0, 0, 7, 0, 0],
[7, 8, 0, 0, 0, 0, 0, 0],
[2, 0, 0, 0, 0, 0, 0, 8],
[0, 0, 7, 0, 0, 0, 8, 0],
[0, 8, 0, 0, 0, 8, 0, 6],
[0, 0, 0, 0, 8, 0, 6, 0]]
g.dijkstra(0)

Floyd
理解
其原理为:假定存在某个结点k在结点i和结点j之间,如果i和j之间的路径长度大于经过k到,则将其距离更新成经过结点k的路径长度。因此在实现中使用三个循环即可,但是要注意循环的顺序,最外层是结点k的遍历,这样可以避免i和j的遍历过早的把最短路径给定下来。

结果为:

实现
import copy
INF = 1e9
class Graph():
def __init__(self, graph):
self.V = len(graph)
self.graph = graph
# 打印距离
def print_distance(self, dist):
for i in range(self.V):
for j in range(self.V):
print('%5s' % (dist[i][j] if dist[i][j] != INF else 'INF'), end='')
print('')
# 核心算法
def floydwarshall(self):
dist = copy.deepcopy(self.graph)
for k in range(self.V):
for i in range(self.V):
for j in range(self.V):
# 不考虑自己指向自己的情况
if i == j:
continue
dist[i][j] = min(dist[i][j], dist[i][k] + dist[k][j])
self.print_distance(dist)
g = Graph([[INF, INF, 1, INF, INF, INF, INF, INF],
[INF, INF, 2, 8, INF, 5, INF, INF],
[9, INF, INF, INF, 2, INF, INF, INF],
[INF, INF, INF, INF, INF, 2, INF, INF],
[INF, INF, INF, INF, INF, INF, INF, 9],
[INF, INF, INF, INF, INF, INF, INF, INF],
[INF, INF, INF, INF, 1, INF, INF, INF],
[INF, INF, INF, INF, INF, 7, 7, INF]])
g.floydwarshall()
最短路径之Dijkstra算法和Floyd-Warshall算法的更多相关文章
- 最短路径—大话Dijkstra算法和Floyd算法
Dijkstra算法 算法描述 1)算法思想:设G=(V,E)是一个带权有向图,把图中顶点集合V分成两组,第一组为已求出最短路径的顶点集合(用S表示,初始时S中只有一个源点,以后每求得一条最短路径 , ...
- 最短路径——Dijkstra算法和Floyd算法
Dijkstra算法概述 Dijkstra算法是由荷兰计算机科学家狄克斯特拉(Dijkstra)于1959 年提出的,因此又叫狄克斯特拉算法.是从一个顶点到其余各顶点的最短路径算法,解决的是有向图(无 ...
- 最短路径Dijkstra算法和Floyd算法整理、
转载自:http://www.cnblogs.com/biyeymyhjob/archive/2012/07/31/2615833.html 最短路径—Dijkstra算法和Floyd算法 Dijks ...
- 【转】最短路径——Dijkstra算法和Floyd算法
[转]最短路径--Dijkstra算法和Floyd算法 标签(空格分隔): 算法 本文是转载,原文在:最短路径-Dijkstra算法和Floyd算法 注意:以下代码 只是描述思路,没有测试过!! Di ...
- 【转载】Dijkstra算法和Floyd算法的正确性证明
说明: 本文仅提供关于两个算法的正确性的证明,不涉及对算法的过程描述和实现细节 本人算法菜鸟一枚,提供的证明仅是自己的思路,不保证正确,仅供参考,若有错误,欢迎拍砖指正 ----------- ...
- Dijkstra算法和Floyd算法的正确性证明
说明: 本文仅提供关于两个算法的正确性的证明,不涉及对算法的过程描述和实现细节 本人算法菜鸟一枚,提供的证明仅是自己的思路,不保证正确,仅供参考,若有错误,欢迎拍砖指正 ------------- ...
- 最短路径—Dijkstra算法和Floyd算法
原文链接:http://www.cnblogs.com/biyeymyhjob/archive/2012/07/31/2615833.html 最后边附有我根据文中Dijkstra算法的描述使用jav ...
- 最短路径—Dijkstra算法和Floyd算法【转】
本文来自博客园的文章:http://www.cnblogs.com/biyeymyhjob/archive/2012/07/31/2615833.html Dijkstra算法 1.定义概览 Dijk ...
- 图的最短路径——dijkstra算法和Floyd算法
dijkstra算法 求某一顶点到其它各个顶点的最短路径:已知某一顶点v0,求它顶点到其它顶点的最短路径,该算法按照最短路径递增的顺序产生一点到其余各顶点的所有最短路径. 对于图G={V,{E}};将 ...
随机推荐
- Coursera 机器学习笔记(三)
主要为第四周.第五周课程内容:神经网络 神经网络模型引入 之前学习的线性回归还是逻辑回归都有个相同缺点就是:特征太多会导致计算量太大.如100个变量,来构建一个非线性模型.即使只采用两两特征组合,都会 ...
- 排序算法 - 选择排序(selection sort)
选择排序(Selection sort)跟插入排序一样,也是O(n^2)的复杂度,这个排序方式也可以用我们的扑克牌来解释. 概念 桌面上有一堆牌,也是杂乱无章的,现在我们想将牌由小到大排序,如果使用选 ...
- canvas学习总结三:绘制路径-线段
Canvas绘图环境中有些属于立即绘制图形方法,有些绘图方法是基于路径的. 立即绘制图形方法仅有两个strokeRect(),fillRect(),虽然strokezText(),fillText() ...
- 单机部署 kubernets 方法汇总
#minikube : 可以方便的在本机用虚拟机创建一个开箱即用的Kubernetes集群 #kubeadm : 可以自动化的将多台Ubuntu或者CentOS主机组建成集群 #nanokube,ki ...
- WCF学习——构建一个简单的WCF应用(二)
我们接着上一篇文章进行讲解 http://www.cnblogs.com/songjianhui/p/7060698.html 一:客户端通过添加引用调用服务 WCF应用服务被成功寄宿后,WCF服务应 ...
- 点击页面其它地方隐藏该div的方法
思路一 第一种思路分两步 第一步:对document的click事件绑定事件处理程序,使其隐藏该div 第二步:对div的click事件绑定事件处理程序,阻止事件冒泡,防止其冒泡到document,而 ...
- 字符串距离 简单DP
字符串距离 时间限制: 1 Sec 内存限制: 128 MB 题目描述 设有字符串 X,我们称在 X 的头尾及中间插入任意多个空格后构成的新字符串为 X 的扩展串,如字符串 X 为"abc ...
- js中的匿名函数自执行
随笔,java中因为有修饰符的存在,那就有private类的存在,js不一样,没有修饰词一说,因此为了防止全局变量的污染,js中就出现了匿名函数,直接上code,看到的人可以自己体会: (functi ...
- hibernate in List查询条件 sum求和使用参考
@Override public Integer getSumZongShuByidList(List<String> idList){ Integer zongshu = 0; Stri ...
- 保存Druid的监控记录
继上篇帖子之后 , 公司又要求将Druid Monitor的监控信息保存起来 , 因为Druid的监控记录在是缓存的,重启之后无法找回,所以需要做持久化,定期把监控记录转存到日志文件中 研究了半天 , ...