图论进阶之路-最短路(Floyd)

时间复杂度:O(n^3)

使用场景:当需要得知任意两个点的最短距离以及其路径时使用

准备:需要两个矩阵

一个记录最短距离(D)

一个记录最短路径的最后一个结点(P)

其核心在于不断的判断越过中间结点是否比不越过中间节点距离更短,迭代的结果也会影响到后面的路径的更新,通过不断的更新,使得每两个节点直接的距离被都更新到最短

具体过程:

1.初始化 D,P 矩阵,D 矩阵初始化为所有结点的入度距离,P 矩阵 初始化为所有结点的入度结点

java 复制代码
        int MAX = Integer.MAX_VALUE;
		int[][] D = {
			{MAX,MAX,MAX,MAX,  6},
			{  9,MAX,  3,MAX,MAX},
			{  2,MAX,MAX,  5,MAX},
			{MAX,MAX,MAX,MAX,  1},
			{MAX,MAX,MAX,MAX,MAX}
		};
		int[][] P = {
			{-1,-1,-1,-1, 0},
			{ 1,-1, 1,-1,-1},
			{ 2,-1,-1, 2,-1},
			{-1,-1,-1,-1, 3},
			{-1,-1,-1,-1,-1}
		};

2.将每一个点都做一次中间结点

3.在当前中间节点的基础上,遍历所有结点,更新最短路

关于两个矩阵更新规则:

  • D: 根据上一次的 D ,若 遍历到的结点到中间结点 + 中间结点到目标结点 < 上一次遍历到的结点到目标结点,就更新
  • P: 若 D 发生变动,则将路径更新为 上一次 中间结点到目标节点的路径

共五个结点,故我们需要重复 5 次 2,3 步骤

java 复制代码
public static void main(String[] args) {
		int MAX = Integer.MAX_VALUE/2;
		int[][] D = {
			{MAX,MAX,MAX,MAX,  6},
			{  9,MAX,  3,MAX,MAX},
			{  2,MAX,MAX,  5,MAX},
			{MAX,MAX,MAX,MAX,  1},
			{MAX,MAX,MAX,MAX,MAX}
		};
		int[][] P = {
			{-1,-1,-1,-1, 0},
			{ 1,-1, 1,-1,-1},
			{ 2,-1,-1, 2,-1},
			{-1,-1,-1,-1, 3},
			{-1,-1,-1,-1,-1}
		};
		for(int k=0;k<5;k++) {//中间结点	
	
			//遍历所有的结点对
			for(int i=0;i<5;i++) {
				for(int j=0;j<5;j++) {
					if(D[i][k] + D[k][j] < D[i][j]) {
						D[i][j] = D[i][k] + D[k][j];
						P[i][j] = P[k][j];
					}
				}
			}
			
		}
	}

当中间点为 0 时,两个矩阵的更新结果为:

[∞, ∞, ∞, ∞, 6]

[9, ∞, 3, ∞, 15]

[2, ∞, ∞, 5, 8]

[∞, ∞, ∞, ∞, 1]

[∞, ∞, ∞, ∞, ∞]


[-1, -1, -1, -1, 0]

[1, -1, 1, -1, 0]

[2, -1, -1, 2, 0]

[-1, -1, -1, -1, 3]

[-1, -1, -1, -1, -1]

=================================

当中间点为 1 时,两个矩阵的更新结果为:

[∞, ∞, ∞, ∞, 6]

[9, ∞, 3, ∞, 15]

[2, ∞, ∞, 5, 8]

[∞, ∞, ∞, ∞, 1]

[∞, ∞, ∞, ∞, ∞]


[-1, -1, -1, -1, 0]

[1, -1, 1, -1, 0]

[2, -1, -1, 2, 0]

[-1, -1, -1, -1, 3]

[-1, -1, -1, -1, -1]

=================================

当中间点为 2 时,两个矩阵的更新结果为:

[∞, ∞, ∞, ∞, 6]

[5, ∞, 3, 8, 11]

[2, ∞, ∞, 5, 8]

[∞, ∞, ∞, ∞, 1]

[∞, ∞, ∞, ∞, ∞]


[-1, -1, -1, -1, 0]

[2, -1, 1, 2, 0]

[2, -1, -1, 2, 0]

[-1, -1, -1, -1, 3]

[-1, -1, -1, -1, -1]

=================================

当中间点为 3 时,两个矩阵的更新结果为:

[∞, ∞, ∞, ∞, 6]

[5, ∞, 3, 8, 9]

[2, ∞, ∞, 5, 6]

[∞, ∞, ∞, ∞, 1]

[∞, ∞, ∞, ∞, ∞]


[-1, -1, -1, -1, 0]

[2, -1, 1, 2, 3]

[2, -1, -1, 2, 3]

[-1, -1, -1, -1, 3]

[-1, -1, -1, -1, -1]

=================================

当中间点为 4 时,两个矩阵的更新结果为:

[∞, ∞, ∞, ∞, 6]

[5, ∞, 3, 8, 9]

[2, ∞, ∞, 5, 6]

[∞, ∞, ∞, ∞, 1]

[∞, ∞, ∞, ∞, ∞]


[-1, -1, -1, -1, 0]

[2, -1, 1, 2, 3]

[2, -1, -1, 2, 3]

[-1, -1, -1, -1, 3]

[-1, -1, -1, -1, -1]

=================================

4.若最后需要得到最短路路径:可以通过 先找到 路径矩阵的位置,得到前一个点,再找到该点与前一个点的前一个点,直到前一个点变成自身为止

如:我们要找到 v1 到 v0 的最短路径

先找到 1 -> 0 的最近的前一个结点,也就是 P[1][0] = 2

得知了前一个结点为 2 ,记录路径 2 -> 0

继续往前找,1 -> 2 的前一个结点,也就是 P[1][2] = 1

得知了前一个结点为 1,记录路径 1 -> 2 -> 0

再继续往前就是寻找 1 -> 1 ,自己找自己的时候就代表路径已经完整了

故 v1 到 v0 的最短路径为: 1 -> 2 -> 0

相关推荐
大数据追光猿4 分钟前
Python应用算法之贪心算法理解和实践
大数据·开发语言·人工智能·python·深度学习·算法·贪心算法
Dream it possible!21 分钟前
LeetCode 热题 100_在排序数组中查找元素的第一个和最后一个位置(65_34_中等_C++)(二分查找)(一次二分查找+挨个搜索;两次二分查找)
c++·算法·leetcode
夏末秋也凉22 分钟前
力扣-回溯-46 全排列
数据结构·算法·leetcode
南宫生22 分钟前
力扣每日一题【算法学习day.132】
java·学习·算法·leetcode
柠石榴27 分钟前
【练习】【回溯No.1】力扣 77. 组合
c++·算法·leetcode·回溯
Leuanghing27 分钟前
【Leetcode】11. 盛最多水的容器
python·算法·leetcode
qy发大财27 分钟前
加油站(力扣134)
算法·leetcode·职场和发展
王老师青少年编程28 分钟前
【GESP C++八级考试考点详细解读】
数据结构·c++·算法·gesp·csp·信奥赛
qy发大财29 分钟前
柠檬水找零(力扣860)
算法·leetcode·职场和发展
瓦力的狗腿子31 分钟前
Starlink卫星动力学系统仿真建模番外篇6-地球敏感器
算法·数学建模·simulink