弗洛伊德算法三重循环的最通俗理解

2023-11-06 17:30

本文主要是介绍弗洛伊德算法三重循环的最通俗理解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

我是标题党

  • 首先本解释仅限无向图无负边权…直奔主题,我先上一段代码
//i是起点,j是终点,k是中转点
for(int k = 1; k <= n; k++){for(int i = 1; i <= n; i++){for(int j = 1; j <= n; j++){if(d[i][j] > d[i][k] + d[k][j]){d[i][j] = d[i][k] + d[k][j];}}}
}
  • 我大一刚接触这玩意,压根不懂,直到大四考研重新接触,理解一小会就明白了。接下来我就给大伙们理一理我的思维。
i和j的关系
  • i是起点,j是终点,大家都懂。然后我给大家一个重要的式子,d[i][j]==d[j][i],在无向图中,明显这是成立的。这不是废话吗? 但是理解的数学(物理)意义大家真正理解了吗?
  • d[i][j]==d[j][i]表示从i点出发到j点的最短距离等于从j点出发到i点的最短距离这不是废话吗? 这里我引出一个理解i和j关系最重要的数学概念 对称性。因此这条式子代表了i和j具有对称性,在线性代数上表示, d n × n d_{n×n} dn×n是对称矩阵。因此可以理解为,i和j可以互相替换。 放到程序里,原程序👇
for(int k = 1; k <= n; k++){for(int i = 1; i <= n; i++){for(int j = 1; j <= n; j++){if(d[i][j] > d[i][k] + d[k][j]){d[i][j] = d[i][k] + d[k][j];}}}
}

完全等价于👇

for(int k = 1; k <= n; k++){for(int j = 1; j <= n; j++){for(int i = 1; i <= n; i++){if(d[j][i] > d[j][k] + d[k][i]){d[j][i] = d[j][k] + d[k][i];}}}
}
  • 重点来了这个👇代表什么数学(物理)含义?
for(int i = 1; i <= n; i++){for(int j = 1; j <= n; j++){//......}
}
  • 说人话就是遍历整个 d n × n d_{n×n} dn×n矩阵,并且要更新它,在程序里面就是,遍历所有以i起点到以j为终点的最短距离。实际上我刚刚也说明,i和j可以互换。所以上面的程序,这关于i和j循环一定要放在一起。(当然不放一起也可以,也有它自己的数学含义,但是这绝对不是弗洛伊德算法)
关于k为什么要放在最前面
  • 我相信这个问题有很多人无法理解。上面我们已经清楚i和j的关系,下面我就当作大家都明白了,不明白的话再看几遍。
  • 我们可以想一想,k在第一层循环,第二、三层循环是在遍历矩阵,所以这个程序的思路是首先固定k点,遍历更新 d n × n d_{n×n} dn×n矩阵,没毛病吧。这里我先举个栗子,看图
    在这里插入图片描述
  • k是啥?k是i到j上的中继点
  • 假设这里k为0,更新的路径为1-0-2,1-0-3,1-0-4,2-0-3,2-0-4,3-0-4
  • 假设接下来更新的k为1,通过k=0已经更新的路径,实际上此时更新路径为2-0-1-3,2-0-1-4,3-0-1-4,2-1-3…看到这里你发现d[2][3]更新了3次,推下去整个k遍历完你发现可以更新所有2->3的路径
  • 大家懂了吧

这篇关于弗洛伊德算法三重循环的最通俗理解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

深入理解Apache Kafka(分布式流处理平台)

《深入理解ApacheKafka(分布式流处理平台)》ApacheKafka作为现代分布式系统中的核心中间件,为构建高吞吐量、低延迟的数据管道提供了强大支持,本文将深入探讨Kafka的核心概念、架构... 目录引言一、Apache Kafka概述1.1 什么是Kafka?1.2 Kafka的核心概念二、Ka

openCV中KNN算法的实现

《openCV中KNN算法的实现》KNN算法是一种简单且常用的分类算法,本文主要介绍了openCV中KNN算法的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录KNN算法流程使用OpenCV实现KNNOpenCV 是一个开源的跨平台计算机视觉库,它提供了各

springboot+dubbo实现时间轮算法

《springboot+dubbo实现时间轮算法》时间轮是一种高效利用线程资源进行批量化调度的算法,本文主要介绍了springboot+dubbo实现时间轮算法,文中通过示例代码介绍的非常详细,对大家... 目录前言一、参数说明二、具体实现1、HashedwheelTimer2、createWheel3、n

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.

Java时间轮调度算法的代码实现

《Java时间轮调度算法的代码实现》时间轮是一种高效的定时调度算法,主要用于管理延时任务或周期性任务,它通过一个环形数组(时间轮)和指针来实现,将大量定时任务分摊到固定的时间槽中,极大地降低了时间复杂... 目录1、简述2、时间轮的原理3. 时间轮的实现步骤3.1 定义时间槽3.2 定义时间轮3.3 使用时

Python循环缓冲区的应用详解

《Python循环缓冲区的应用详解》循环缓冲区是一个线性缓冲区,逻辑上被视为一个循环的结构,本文主要为大家介绍了Python中循环缓冲区的相关应用,有兴趣的小伙伴可以了解一下... 目录什么是循环缓冲区循环缓冲区的结构python中的循环缓冲区实现运行循环缓冲区循环缓冲区的优势应用案例Python中的实现库

如何通过Golang的container/list实现LRU缓存算法

《如何通过Golang的container/list实现LRU缓存算法》文章介绍了Go语言中container/list包实现的双向链表,并探讨了如何使用链表实现LRU缓存,LRU缓存通过维护一个双向... 目录力扣:146. LRU 缓存主要结构 List 和 Element常用方法1. 初始化链表2.

Java嵌套for循环优化方案分享

《Java嵌套for循环优化方案分享》介绍了Java中嵌套for循环的优化方法,包括减少循环次数、合并循环、使用更高效的数据结构、并行处理、预处理和缓存、算法优化、尽量减少对象创建以及本地变量优化,通... 目录Java 嵌套 for 循环优化方案1. 减少循环次数2. 合并循环3. 使用更高效的数据结构4

golang字符串匹配算法解读

《golang字符串匹配算法解读》文章介绍了字符串匹配算法的原理,特别是Knuth-Morris-Pratt(KMP)算法,该算法通过构建模式串的前缀表来减少匹配时的不必要的字符比较,从而提高效率,在... 目录简介KMP实现代码总结简介字符串匹配算法主要用于在一个较长的文本串中查找一个较短的字符串(称为