热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

最短路径问题的几种算法

Floyd算法使用条件可以求出多源最短路,可以处理负权边的情况,但是不能出现负环。时间复杂度O(n3)讲解Floyed算法使用的是动态规划的方法。我们首先观察上图。我们来想一想,根
Floyd算法

使用条件

可以求出多源最短路,可以处理负权边的情况,但是不能出现负环。

时间复杂度

O(n3

讲解

Floyed算法使用的是动态规划的方法。

技术图片

       我们首先观察上图。

       我们来想一想,根据我们以往的经验,如果要让任意两点(例如从顶点a点到顶点b)之间的路程变短,只能引入第三个点(顶点k),并通过这个顶点k中转即a->k->b,才可能缩短原来从顶点a点到顶点b的路程。那么这个中转的顶点k是1~n中的哪个点呢?甚至有时候不只通过一个点,而是经过两个点或者更多点中转会更短,即a->k1->k2b->或者a->k1->k2…->k->i…->b。比如上图中从4号城市到3号城市(4->3)的路程e[4][3]原本是12。如果只通过1号城市中转(4->1->3),路程将缩短为11(e[4][1]+e[1][3]=5+6=11)。其实1号城市到3号城市也可以通过2号城市中转,使得1号到3号城市的路程缩短为5(e[1][2]+e[2][3]=2+3=5)。所以如果同时经过1号和2号两个城市中转的话,从4号城市到3号城市的路程会进一步缩短为10。通过这个的例子,我们发现每个顶点都有可能使得另外两个顶点之间的路程变短。好,下面我们将这个问题一般化。

       当任意两点之间不允许经过第三个点时,这些城市之间最短路程就是初始路程,如下。

技术图片

 我们如果在每两个点之间设置一个中转点,看看是使用这个中转点之前路程更短还是使用之后路程更短,而两个点到中转点的距离都已经是最短路程,这样最后计算出来的就一定是最短路程了。

只需要使用最简单粗暴的做法,将出发点、结束点、中转点都枚举一遍就可以了。

状态转移方程:

d[i][j]=min(d[i][k]+d[k][j],d[i][j])

这样,再写出Floyd算法的核心代码就很容易了。

另外需要注意的是:Floyd算法不能解决带有“负权回路”(或者叫“负权环”)的图,因为带有“负权回路”的图没有最短路。例如下面这个图就不存在1号顶点到3号顶点的最短路径。因为1->2->3->1->2->3->…->1->2->3这样路径中,每绕一次1->-2>3这样的环,最短路就会减少1,永远找不到最短路。其实如果一个图中带有“负权回路”那么这个图则没有最短路。

技术图片

核心代码

for(k=1;k<=n;k++) //枚举中转点
    for(i=1;i<=n;i++) //枚举起点        
        for(j=1;j<=n;j++)          //枚举终点
            d[i][j]=min(d[i][k]+d[k][j],d[i][j]);
Dijkstra算法

使用条件

求单源最短路径,不能处理负权。

时间复杂度

O(n2

讲解

Dijkstra算法使用的是贪心方法,d[i]表示起点s0到i的最短距离。

从起点s0开始,选择未访问过的离s0最近的一个点i,也就是最小的d[i],因为所以边权为正,不会存在更短的路径到达i,保证了贪心的正确性。然后将i作为中间点,更新经过i可到达的点的最短路距离,继续贪心寻找未访问过的最近的一个点,经过n次贪心,算法结束。

看图:

技术图片技术图片

根据这个图,Dijkstra算法应该就很好理解了。

核心代码

for (i = 1; k <= n; k++)
{
    maxn = 0x7fffffff;
    for (j = 1; j <= n; j++)                 //找出未访问最小的d[j]
    {
        if (!vis[j] && d[j] < maxn)
        {
            maxn = d[j];
            k = i;
        }

    }
    vis[k] = 1;
    for (j = 1; j <= n; j++)          //k作为中间点,更新起点经过k到达其他点的d[j]
        if (w[k][j])
        {
            d[j] = min{ d[k] + w[k][j],d[j] };
        }
}
SPFA算法

使用条件

求单源最短路,可以处理负权边

时间复杂度

对于稀疏图,为O(km),k为较小的常数,而对于稠密图或者构造的网格图,会提高到O(n*m)

讲解

建立一个队列,初始时队列里只有起始点,在建立一个表格记录起始点到所有点的最短路径(该表格的初始值要赋为极大值,该点到他本身的路径赋为0)。然后执行松弛操作,用队列里有的点去刷新起始点到所有点的最短路,如果刷新成功且被刷新点不在队列中则把该点加入到队列最后。重复执行直到队列为空。

图:

技术图片

源点A首先入队,并且AB松弛

技术图片

扩展与A相连的边,B,C 入队并松弛。

技术图片

B,C分别开始扩展,D入队并松弛

技术图片

D出队,E入队并松弛。

技术图片

E出队,此时队列为空,源点到所有点的最短路已被找到,A->E的最短路即为8

技术图片

以上就是SPFA算法的过程。

核心代码

q.push(s);
vis[s]=1;  //源点s入队,标记入队
while(q.size())
{
       u=q.front();q.pop();vis[u]=0;        //取出队头,标记未入队
       for(i=head[u];i;i=next[i])
       {
              v=ver[i];
              w=edge[i];
              if(dis[u]+w<dis[v])
              {
                     dis[v]=dis[u]+w;
                     if(!vis[v])   {q.push(v);vis[v]=1;}    //如果没有在队列,入队,标记已入队
              }    
       }
}

最短路径问题的几种算法


推荐阅读
  • 嵌入式开发环境搭建与文件传输指南
    本文详细介绍了如何为嵌入式应用开发搭建必要的软硬件环境,并提供了通过串口和网线两种方式将文件传输到开发板的具体步骤。适合Linux开发初学者参考。 ... [详细]
  • 解决TensorFlow CPU版本安装中的依赖问题
    本文记录了在安装CPU版本的TensorFlow过程中遇到的依赖问题及解决方案,特别是numpy版本不匹配和动态链接库(DLL)错误。通过详细的步骤说明和专业建议,帮助读者顺利安装并使用TensorFlow。 ... [详细]
  • 鼠标悬停出现提示信息怎么做
    概述–提示:指启示,提起注意或给予提醒和解释。在excel中会经常用到给某个格子增加提醒信息,比如金额提示输入数值或最大长度值等等。设置方式也有多种,简单的,仅为单元格插入批注就可 ... [详细]
  • 深入解析动态代理模式:23种设计模式之三
    在设计模式中,动态代理模式是应用最为广泛的一种代理模式。它允许我们在运行时动态创建代理对象,并在调用方法时进行增强处理。本文将详细介绍动态代理的实现机制及其应用场景。 ... [详细]
  • 如何清除Chrome浏览器地址栏的特定历史记录
    在使用Chrome浏览器时,你可能会发现地址栏保存了大量浏览记录。有时你可能希望删除某些特定的历史记录而不影响其他数据。本文将详细介绍如何单独删除地址栏中的特定记录以及批量清除所有历史记录的方法。 ... [详细]
  • 利用Selenium与ChromeDriver实现豆瓣网页全屏截图
    本文介绍了一种使用Selenium和ChromeDriver结合Python代码,轻松实现对豆瓣网站进行完整页面截图的方法。该方法不仅简单易行,而且解决了新版Selenium不再支持PhantomJS的问题。 ... [详细]
  • 探索新一代API文档工具,告别Swagger的繁琐
    对于后端开发者而言,编写和维护API文档既繁琐又不可或缺。本文将介绍一款全新的API文档工具,帮助团队更高效地协作,简化API文档生成流程。 ... [详细]
  • 本文探讨了在构建应用程序时,如何对不同类型的数据进行结构化设计。主要分为三类:全局配置、用户个人设置和用户关系链。每种类型的数据都有其独特的用途和应用场景,合理规划这些数据结构有助于提升用户体验和系统的可维护性。 ... [详细]
  • Linux中的yum安装软件
    yum俗称大黄狗作用:解决安装软件包的依赖关系当安装依赖关系的软件包时,会将依赖的软件包一起安装。本地yum:需要yum源,光驱挂载。yum源:(刚开始查看yum源中的内容就是上图 ... [详细]
  • 气象对比分析
    本文探讨了不同地区和时间段的天气模式,通过详细的图表和数据分析,揭示了气候变化的趋势及其对环境和社会的影响。 ... [详细]
  • 探讨 HDU 1536 题目,即 S-Nim 游戏的博弈策略。通过 SG 函数分析游戏胜负的关键,并介绍如何编程实现解决方案。 ... [详细]
  • 实用正则表达式有哪些
    小编给大家分享一下实用正则表达式有哪些,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下 ... [详细]
  • 本文探讨了如何利用NFC技术,将存储在Android手机中的患者信息安全高效地传输到台式计算机。重点介绍了适用于医院场景的NFC USB读卡器(如ACR122U)的应用方法。 ... [详细]
  • 深入理解K近邻分类算法:机器学习100天系列(26)
    本文详细介绍了K近邻分类算法的理论基础,探讨其工作原理、应用场景以及潜在的局限性。作为机器学习100天系列的一部分,旨在为读者提供全面且深入的理解。 ... [详细]
  • 本文回顾了2017年的转型和2018年的收获,分享了几家知名互联网公司提供的工作机会及面试体验。 ... [详细]
author-avatar
手机用户2602884633
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有