我的编程空间,编程开发者的网络收藏夹
学习永远不晚

PHP数据结构-图的应用:最短路径的示例分析

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

PHP数据结构-图的应用:最短路径的示例分析

这篇文章主要介绍了PHP数据结构-图的应用:最短路径的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。

什么是最短路径

今天我们学习的是图的应用中另外一个经典的问题,也就是 最短路径 的问题。这个问题和最小生成树是不同的,最小生成树的要求是要连通所有的结点,并且走得是权值最小的那条路线。而最短路径则是指的从某个顶点到另一个顶点中权值最小的那条路径。这条路径不一定是包含在最小生成树中的,所以它们并没有太大的联系。

PHP数据结构-图的应用:最短路径的示例分析

从这张图来看,我们从结点 1 到结点 2 的最短路径是 2 ,这个很明显。那么从结点 1 到结点 3 呢?可不是直接的中间那个权值为 6 的路径,而是走 1->2->3 这条路径,也就是权值加起来为 5 的这条路径。

然后我们再来看结点 3 ,它到结点 1 最短路径应该是走 3->4->1 这条路径,也就是权值为 6 的这条路径,而不是中间的那条直线的权值为 7 的路径。

没错,这就是最短路径的概念了。在最短路径中,我们一般会解决单向图的问题,但实际生活中呢?最典型的地图相关的应用其实是都是双向图的。不过这并不影响我们的学习,我们可以把这个示例图中的结点看成是城市火车站点,就算是连接结点 1 和结点 3 的火车线路,也不一定来去的时间都是相同的。比如说从长沙到北京的 Z2 次火车全部运行时间为14小时42分,而回来的 Z1 次则是14小时10分。那么我们是否可以选择其它的火车,比如有趟火车从长沙到石家庄可能只需要8小时,然后从石家庄到北京只需要2小时,这样我们选择这条线路的总时间就只需要10小时了(当然,这只是例子,大家在非高铁的情况下肯定还是更多地会选择起始站的火车来坐)。

多源最短路径 Floyd 算法

首先,我们先说一个多源最短路径的算法。那么什么叫做多源呢?

其实就是这一个算法就能够得出所有结点到所有结点之间的最短路径。没错,就这一个算法,不管哪个结点到哪个结点,它们之间的最短路径都一次性算出来了。神奇吗?不不不,更神奇的,而且你一会就会叫出 Oh!My God! 的是它的核心代码,只有五行!!

function Floyd($graphArr){    $n = count($graphArr);        for($k = 1;$k<=$n;$k++){ // 设 k 为经过的结点        for($i = 1;$i<=$n;$i++){            for($j = 1;$j<=$n;$j++){                // 如果经过 k 结点 能使 i 到 j 的路径变短,那么将 i 到 j 之间的更新为通过 k 中转之后的结果                 if($graphArr[$i][$j] > $graphArr[$i][$k] + $graphArr[$k][$j]){                    $graphArr[$i][$j] = $graphArr[$i][$k] + $graphArr[$k][$j];                }            }        }    }    for($i = 1;$i<=$n;$i++){        for($j = 1;$j<=$n;$j++){            echo $graphArr[$i][$j], ' ';        }        echo PHP_EOL;    }}// 请输入结点数:4 // 请输入边数:8// 请输入边,格式为 出 入 权:1 2 2// 请输入边,格式为 出 入 权:1 3 6// 请输入边,格式为 出 入 权:1 4 4 // 请输入边,格式为 出 入 权:2 3 3// 请输入边,格式为 出 入 权:3 1 7// 请输入边,格式为 出 入 权:3 4 1// 请输入边,格式为 出 入 权:4 1 5// 请输入边,格式为 出 入 权:4 3 12// 0 2 5 4 // 9 0 3 4 // 6 8 0 1 // 5 7 10 0

我们可以先验证下结果,就是注释中最后输出的矩阵。结点 1 到结点 2、3、4的最短距离为 2 、5 、4 。结点 3 到结点 1 、2 、4 的最短距离为 6 、8 、1 。也就是说,原来的那个图的邻接矩阵成了这个最短路径的矩阵。每一行代表每个结点到其它结点的最短距离。

好吧,结果没问题,那么代码到底是写得啥玩意?这个 k 是什么?别急,我们一步一步来看。

假设两点之间的距离不是最短的,那么肯定是有另外一个点做为媒介进行跳转,由 i 点先跳到这个点然后再跳向 j 点,这样的一条路径是比直接的 i 到 j 要近的,我们就定义这个点为 k 点

但是我们不知道要走哪个结点呀,而且还有可能不只是一个 k ,或许我们从 i 到 j 要经历好多个 k ,这时候,我们就从 k 开始遍历,也就是第一层循环

在第一层循环下,进行我们正常的 i 和 j 的遍历循环,获得 i 直接到 j 的长度,也就是 [i][j] 。这时,由于有最外层的 k 存在,所以我们也知道了如果 i 从 k 走再从 k 到 j 的长度,也就是 [i][k] + [k][i] 这段距离

很明显,如果 [i][k] + [k][i] 的距离要比 [i][j] 短的话,更新 [i][j] 的值为 [i][k] + [k][i]

内部的 i 和 j 循环完成后,第 1 个结点做为媒介跳转的遍历也完成了,当前的矩阵中各个结点之间的权重已经是经过第 1 个结点做为媒介之后的最短路径了

但是呢,这并不准确,说不定我们可能经过 i 、k1 、 k2 、 j 的路径才是最短的,所以外层的 k 循环继续遍历并将第 2 个结点作为媒介结点

循环往复直到所有结点都做过一次中间的媒介结点之后,我们就得到了一个最短路径的矩阵图,也就是我们上面测试代码中输出的结果

我们就拿结点 4 和结点 3 来说明。我们定义 4 为 i ,结点 3 为 j 。

初始化时,[i][j] 为 12 ,这个没什么问题,单向图的那条带箭头的边的权值就是 12 。

然后当 k 为 1 时,也就是我们经过结点 1 来看路径有没有变短,这时 [i][k] 是 5 ,[k][j] 是 6 ,OK,路径变成 11 了,把 [i][j] 的值改成 11 。

同时,在 i 为 4 ,j 为 2 的情况下,他们两个的最短路径也在这次 k=1 的循环中被赋值为 7 。最开始 4 到 2 是没有直接的边的,现在在结点 1 的连接下,他们有了路径,也就是 [4][2] = [4][1] + [1][2] = 7 。

当 k 为 2 时,[i][j] 为 11 ,这时 [i][k] 就是上面说过的 [4][2] 。也就是 7 ,而 [k][j] 则是 3 ,路径又缩小了,[i][k] + [k][j] = 10 ,[i][j] 现在又变成了 10 。

循环继续,但已经没有比这条路径更小的值了,所以最后 [4][2] 的最短路径就是 10 。

看着晕吗?拿出笔来在纸上或者本子上自己画画,每一步的 k 都去画一下当前的最短路径矩阵变成什么样了。这样画一次之后,马上就知道这个 Floyd 算法的核心奥秘所在了。

不得不说,前人的智慧真的很伟大吧,不过说是前人,其实 Floyd 大佬在 1962 年才发表了这个算法,但这个算法的核心思想却是数学中的动态规划的思想。所以说,算法和数学是没法分家的,各位大佬哪个不是数学界的一把手呢。

单源最短路径 Dijkstra 算法

说完了多源最短路径,我们再讲一个鼎鼎大名的单源最短路径的算法。虽说上面的多源很牛X,但是它的时间复杂度也就是时间效率也确实是太差了,没看错的话三个 N 次的循环嵌套就是 O(N3)。如果数据稍微多一点的话基本就可以从 Oh!My God! 变成 Oh!FxxK! 了。而且大多数情况下,我们的需求都会是固定的求从某一点到另一点的最短路径问题,也就是单源最短路径问题。这时,就可以使用这种效率稍微好一点的算法来快速地解决了。

// origin 表示源点,也就是我们要看哪个结点到其它结点的最短路径function Dijkstra($graphArr, $origin){    $n = count($graphArr);    $dis = []; // 记录最小值    $book = []; // 记录结点是否访问过    // 初始化源点到每个点的权值    for ($i = 1; $i <= $n; $i++) {        $dis[$i] = $graphArr[$origin][$i]; // 源点到其它点的默认权值        $book[$i] = 0; // 所有结点都没访问过    }    $book[$origin] = 1; // 源点自身标记为已访问    // 核心算法    for ($i = 1; $i <= $n - 1; $i++) {        $min = INFINITY;        // 找到离目标结点最近的结点        for ($j = 1; $j <= $n; $j++) {            // 如果结点没有被访问过,并且当前结点的权值小于 min 值            if ($book[$j] == 0 && $dis[$j] < $min) {                $min = $dis[$j]; // min 修改为当前这个节点的路径值                $u = $j; // 变量 u 变为当前这个结点            }            // 遍历完所有结点,u 就是最近的那个顶点        }        $book[$u] = 1; // 标记 u 为已访问        for ($v = 1; $v <= $n; $v++) {            // 如果 [u][v] 顶点小于无穷            if ($graphArr[$u][$v] < INFINITY) {                // 如果当前 dis[v] 中的权值大于 dis[u]+g[u][v]                if ($dis[$v] > $dis[$u] + $graphArr[$u][$v]) {                    // 将当前的 dis[v] 赋值为 dis[u]+g[u][v]                    $dis[$v] = $dis[$u] + $graphArr[$u][$v];                }            }        }        // 最近的结点完成,继续下一个最近的结点    }    for ($i = 1; $i <= $n; $i++) {        echo $dis[$i], PHP_EOL;    }}// 请输入结点数:4 // 请输入边数:8// 请输入边,格式为 出 入 权:1 2 2// 请输入边,格式为 出 入 权:1 3 6// 请输入边,格式为 出 入 权:1 4 4 // 请输入边,格式为 出 入 权:2 3 3// 请输入边,格式为 出 入 权:3 1 7// 请输入边,格式为 出 入 权:3 4 1// 请输入边,格式为 出 入 权:4 1 5// 请输入边,格式为 出 入 权:4 3 12// 测试第四个结点到其它结点的最短路径Dijkstra($graphArr, 4);// 5// 7// 10// 0

代码一下增加了不少吧,不过仔细看一下核心的算法部分,这次只是两层循环的嵌套了,时间复杂度一下子就降到了 O(N2) ,这一下就比 Floyd 算法提升了很多。当然,它的场景也是有限的,那就是只能一个结点一个结点的计算。

好了,我们还是来看一下 Dijkstra 到底在干嘛吧。我们依然是使用上面那个简单的图,并且还是研究结点 4 到其它结点的算法执行情况。

首先,我们初始化结点 4 到其他所有结点的默认值,这时结点 4 到结点 2 是没有直接路径的,所以是无穷大,而到结点 1 是 5,到结点 3 是 12 。

然后将结点 4 标记为已访问,也就是 book[4] = 1

进入核心算法,从头开始遍历结点,这里是标记为 i 下标,因为这里是单源的最短路径,所以我们不需要再看自己到自己的最短路径了,只需要 n-1 次循环就可以了

开始 j 层的循环,先判断当前的结点是否已经被标记过,没有被标记过的话再看它的值是否是最小的,最后循环完成后获得一个从结点 4 出发的权值最小的路径,并将这条路径到达的结点下标记为 u ,标记 u 下标的这个结点为已访问结点

进入 v 循环,判断图中 u 到 v 的结点是否是无穷,如果不是的话再判断 u 到 v 的结点加上原来的 dis[u] 的权值是否小于 dis[v] 中记录的权值,如果比这个小的话,更新 dis[v] 为 u 到 v 的结点加上原来的 dis[u] 的权值

循环重复地进行比较完成算法

对于结点 4 来说,dis 经历了如下的变化:

首先,默认情况下 dis = [5, 9999999, 12, 0]

第一次循环后,结点1 完成查找,并在 v 的循环中发现了可以从结点1 到结点2 和结点3 而且比原来的值都要小 ,于是 dis = [5, 7, 11, 0]

第二次循环后,结点2 完成查找,这次循环发现从结点2 到结点3 的距离更短,于是 dis = [5, 7, 10, 0]

第三次循环后,结点3 完成查找,没有发现更短的路径,dis = [5, 7, 10, 0]

看明白了吗?不明白的话自己试试吧,不管是断点还是在中间输出一下 dis 和 book ,都能够帮助我们更好地理解这个算法的每一步是如何执行的。从代码中就可以看出来,这个 Dijkstra 算法的时间复杂度是 O(N2) ,这可比 Floyd 快了不少了吧。

感谢你能够认真阅读完这篇文章,希望小编分享的“PHP数据结构-图的应用:最短路径的示例分析”这篇文章对大家有帮助,同时也希望大家多多支持编程网,关注编程网行业资讯频道,更多相关知识等着你来学习!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

PHP数据结构-图的应用:最短路径的示例分析

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

PHP数据结构-图的应用:最短路径的示例分析

这篇文章主要介绍了PHP数据结构-图的应用:最短路径的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。什么是最短路径今天我们学习的是图的应用中另外一个经典的问题,也就是
2023-06-20

python中最短路径问题的示例分析

小编给大家分享一下python中最短路径问题的示例分析,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!说明1、最短路径问题是图论研究中的经典算法问题,用于计算从一个顶点到另一个顶点的最短路径。2、最短路径问题有几种形式:确定
2023-06-20

PHP数据结构之图存储结构的示例分析

这篇文章主要介绍PHP数据结构之图存储结构的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!图的存储结构图的概念介绍得差不多了,大家可以消化消化再继续学习后面的内容。如果没有什么问题的话,我们就继续学习接下来的
2023-06-20

PHP数据结构中图遍历的示例分析

这篇文章将为大家详细讲解有关PHP数据结构中图遍历的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。图的遍历:深度优先与广度优先树的遍历演化到图的遍历还记得在树的学习中,我们讲到过先序、中序、后序以
2023-06-20

Java数据结构中图的示例分析

这篇文章给大家分享的是有关Java数据结构中图的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。有向图有向图的定义及相关术语定义∶ 有向图是一副具有方向性的图,是由一组顶点和一组有方向的边组成的,每条方向的
2023-06-29

python数据结构堆的示例分析

小编给大家分享一下python数据结构堆的示例分析,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!1、说明堆是用数据结构来实现的一种算法:树,数组均可。堆本身是一棵完全二叉树。2、特点最大堆:所有父节点的值大于子节点的值最小
2023-06-15

Java中数据结构的示例分析

这篇文章将为大家详细讲解有关Java中数据结构的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。1.1.1. 增量内存分配 ArrayList 、 HashMap 、 Vector 等类
2023-06-03

Python Pandas数据结构的示例分析

这篇文章将为大家详细讲解有关Python Pandas数据结构的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。1 Pandas介绍2008年WesMcKinney开发出的库专门用于数据挖掘的开源p
2023-06-29

C++数据结构中list的示例分析

小编给大家分享一下C++数据结构中list的示例分析,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!前言list相较于vector来说会显得复杂,它的好处是在任意位置插入,删除都是一个O(1)的时间复杂度。一、list的节点
2023-06-25

java数据结构之树的示例分析

这篇文章主要介绍java数据结构之树的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!树定义和基本术语定义树(Tree)是n(n≥0)个结点的有限集T,并且当n>0时满足下列条件:(1)有且仅有一个特定的称为根
2023-05-30

python数据结构算法的示例分析

小编给大家分享一下python数据结构算法的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!1.算法分析的定义有这样一个问题:当两个看上去不同的程序 解决同
2023-06-22

Python数据结构创建的示例分析

本篇文章为大家展示了Python数据结构创建的示例分析,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。1. 列表list:变量赋值方式:shoplist = [apple, mango, carrot
2023-06-17

C++数据结构红黑树的示例分析

这篇文章给大家分享的是有关C++数据结构红黑树的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。概念和性质红黑树的概念: 红黑树,是一种二叉搜索树,但在每个结点上增加一个存储位表示结点的颜色,可以是Red或
2023-06-29

JDK 1.8中HashMap数据结构的示例分析

这篇文章给大家分享的是有关JDK 1.8中HashMap数据结构的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。概述JDK 1.8对HashMap361天恒平台制作,进行了比较大的优化,底层实现由之前的“
2023-06-04

Java数据结构与算法的示例分析

这篇文章给大家分享的是有关Java数据结构与算法的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。第1章 数据结构与算法基础概述1.1 数据结构和算法的重要性算法是程序的灵魂,优秀的程序可以在海量数据计算时
2023-06-29

Java数据结构之链表的示例分析

小编给大家分享一下Java数据结构之链表的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!一、链表的介绍什么是链表链表是一种物理存储单元上非连续、非顺序的存
2023-06-15

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录