图解最短路径之迪杰斯特拉算法(Java实现)

图解最短路径之迪杰斯特拉算法(Java实现)概述迪杰斯特拉算法是由荷兰计算机科学家狄克斯特拉于 1959 年提出的 因此又叫狄克斯特拉算法 是从一个顶点到其余各顶点的最短路径算法 解决的是有权图中最短路径问题 迪杰斯特拉算法主要特点是以起始点为中心向外层层扩展 直到扩展到终点为止 迪杰斯特拉算法采用的是贪心策略 将 Graph 中的节点集分为最短路径计算完成的节点集 S 和未计算完成的节点集 T 每次将从 T 中挑选 V0 gt Vt 最小的节点 Vt 加入

概述

迪杰斯特拉算法是由荷兰计算机科学家狄克斯特拉于1959 年提出的,因此又叫狄克斯特拉算法。是从一个顶点到其余各顶点的最短路径算法,解决的是有权图中最短路径问题。迪杰斯特拉算法主要特点是以起始点为中心向外层层扩展,直到扩展到终点为止。迪杰斯特拉算法采用的是贪心策略,将Graph中的节点集分为最短路径计算完成的节点集S和未计算完成的节点集T,每次将从T中挑选V0->Vt最小的节点Vt加入S,并更新V0经由Vt到T中剩余节点的更短距离,直到T中的节点全部加入S中,它贪心就贪心在每次都选择一个距离源点最近的节点加入最短路径节点集合。迪杰斯特拉算法只支持非负权图,它计算的是单源最短路径,即单个源点到剩余节点的最短路径,时间复杂度为O(n²)。

算法描述

在有向图 G=(V,E) 中,假设每条边 E[i] 的长度为 w[i],找到由顶点 V0 到其余各点的最短值。

算法流程

本节将对算法流程进行模拟,设置Graph为包含7个顶点和9条边的有向无环图,源点为0,计算从源点0到剩余节点的最短路径,Graph如下:

图解最短路径之迪杰斯特拉算法(Java实现)

每个节点将维护shortest和visited两个数据结构,shortest存储v0到该节点的最短路径,visited存储v0到该节点的最短路径是否求出。S为已求出最短路径的节点,T为未求出最短路径的节点。源节点只允许将S中的节点作为中间节点来计算到达其它节点的最短路径,不允许将T中的节点作为中间节点来计算到达其它节点的最短路径。随着S中节点的增加,源节点可达的节点才会增加。初始状态下,源节点只可达节点1和节点3。

算法步骤如下:

1、将源节点(即节点0)加入S中,对shortest和visited数组进行更新。

图解最短路径之迪杰斯特拉算法(Java实现)

2、S中现有节点0,源节点可达T中的节点1和节点3,节点0->节点1距离为6,节点0->节点3距离为2,按距离从小到大排序,因此选择将节点3加入S中。更新源点将节点3作为中间节点到达其它节点的距离。

图解最短路径之迪杰斯特拉算法(Java实现)

3、S中现有节点0和节点3,源节点可达T中的节点1和4,节点0->节点1距离为6,节点0->节点4距离为7,按距离从小到大排序,因此选择将节点1加入S中。更新源点将节点1作为中间节点到达其它节点的距离。

图解最短路径之迪杰斯特拉算法(Java实现)

4、S中现有节点0、1、3,源节点可达T中的节点2、4、5,0->2距离为11,0->4距离为7,0->5距离为9,按距离从小到大排序,因此选择将节点4加入S中。更新源点将节点4作为中间节点到达其它节点的距离。

图解最短路径之迪杰斯特拉算法(Java实现)

5、S中现有节点0、1、3、4,源节点可达T中的节点2、5、6,0->2距离为11,0->5距离为9,0->6距离为8,按距离从小到大排序,因此选择将节点6加入S中。更新源点将节点6作为中间节点到达其它节点的距离。

图解最短路径之迪杰斯特拉算法(Java实现)

6、S中现有节点0、1、3、4、6,源节点可达T中的节点2、5,0->2距离为11,0->5距离为9,按距离从小到大排序,因此选择将节点5加入S中。更新源点将节点5作为中间节点到达其它节点的距离。

图解最短路径之迪杰斯特拉算法(Java实现)

7、T中只剩下节点2,0->2距离为11,将节点2加入S中。

图解最短路径之迪杰斯特拉算法(Java实现)

8、算法结束,源点到其它节点的最短路径都已依次求出。

图解最短路径之迪杰斯特拉算法(Java实现)

算法实现

public class DijstraAlgorithm { //不能设置为Integer.MAX_VALUE,否则两个Integer.MAX_VALUE相加会溢出导致出现负权 public static int MaxValue = ; public static void main(String[] args) { Scanner input = new Scanner(System.in); System.out.println("请输入顶点数和边数:"); //顶点数 int vertex = input.nextInt(); //边数 int edge = input.nextInt(); int[][] matrix = new int[vertex][vertex]; //初始化邻接矩阵 for (int i = 0; i < vertex; i++) { for (int j = 0; j < vertex; j++) { matrix[i][j] = MaxValue; } } for (int i = 0; i < edge; i++) { System.out.println("请输入第" + (i + 1) + "条边与其权值:"); int source = input.nextInt(); int target = input.nextInt(); int weight = input.nextInt(); matrix[source][target] = weight; } //单源最短路径,源点 int source = input.nextInt(); //调用dijstra算法计算最短路径 dijstra(matrix, source); } public static void dijstra(int[][] matrix, int source) { //最短路径长度 int[] shortest = new int[matrix.length]; //判断该点的最短路径是否求出 int[] visited = new int[matrix.length]; //存储输出路径 String[] path = new String[matrix.length]; //初始化输出路径 for (int i = 0; i < matrix.length; i++) { path[i] = new String(source + "->" + i); } //初始化源节点 shortest[source] = 0; visited[source] = 1; for (int i = 1; i < matrix.length; i++) { int min = Integer.MAX_VALUE; int index = -1; for (int j = 0; j < matrix.length; j++) { //已经求出最短路径的节点不需要再加入计算并判断加入节点后是否存在更短路径 if (visited[j] == 0 && matrix[source][j] < min) { min = matrix[source][j]; index = j; } } //更新最短路径 shortest[index] = min; visited[index] = 1; //更新从index跳到其它节点的较短路径 for (int m = 0; m < matrix.length; m++) { if (visited[m] == 0 && matrix[source][index] + matrix[index][m] < matrix[source][m]) { matrix[source][m] = matrix[source][index] + matrix[index][m]; path[m] = path[index] + "->" + m; } } } //打印最短路径 for (int i = 0; i < matrix.length; i++) { if (i != source) { if (shortest[i] == MaxValue) { System.out.println(source + "到" + i + "不可达"); } else { System.out.println(source + "到" + i + "的最短路径为:" + path[i] + ",最短距离是:" + shortest[i]); } } } } }

样例输入:

7 10

0 1 6

1 2 5

0 3 2

3 1 7

3 4 5

1 2 5

1 5 3

4 5 5

5 4 2

4 6 1

0

Q&A

问:为什么迪杰斯特拉算法只支持非负权的图?

答:迪杰斯特拉采用的贪心策略,S集合中是已经计算出最短路径的节点,T集合中是未计算出最短路径的节点。假设存在负权,源点为A,已经计算出A->B的最短路径为m,若下一次将C添加进已计算出最短路径的节点集合,而A->C=m,C->B=-1,则会出现A->B的更短路径A->C->B,但迪杰斯特拉不会对已经计算出最短路径的节点重新计算,因此无法更新最短路径,即负权的出现导致无法保证S中节点计算的是最短路径,已经固定dis的点可能会被其它dis大于它的点更新。

问:为什么在代码实现中不能将节点之间不可达用Integer.MAX_VALUE代表?

答:因为两个Integer.MAX_VALUE相加会溢出导致出现负权,所以最好设置为一个比较大且不容易相加溢出的数。

问:迪杰斯特拉算法适用于什么场景?

答:在有些算法书上说,迪杰斯特拉适用于DAG(有向无环图)。但是个人觉得,它所谓的“适用于”,或许只是说可以在DAG上使用,并不代表无向图不能使用,也不能代表有环图不能使用。从迪杰斯特拉的算法原理上来说,无向图是没有问题的,只需要给matrix[source][target]和matrix[target][source]赋上相同的权值,因为它每次只会根据到源点的距离,选取距离最近的一个节点加入,所以有没有方向都无所谓,算法只关注可达点的距离;至于有环图,它对每个节点的距离计算只用了一层遍历去做,并不会陷入死循环,也不会出现重复计算的问题。因此迪杰斯特拉算法是可以用在无向图和有环图中的,适合于求单源最短路径。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/206539.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月19日 下午3:28
下一篇 2026年3月19日 下午3:28


相关推荐

  • potplayer 64位 官网下载地址

    potplayer 64位 官网下载地址potplayer64位最新官网下载地址链接:https://gsf-fl.softonic.com/f94/2ec/be06ec8a283bee92cda70eb3ede9c661d0/PotPlayerSetup64.exe?Expires=1585679112&Signature=cff21a190f0fcb9c61356f8eca9b8155ed6afc48&ur…

    2022年7月12日
    54
  • 谷歌地球手机版2021登不上服务器_谷歌地球连不上服务器是怎么回事

    谷歌地球手机版2021登不上服务器_谷歌地球连不上服务器是怎么回事1、安装运行谷歌地球专业版(GoogleEarthProv7.3)。2、安装运行国家法律允许使用的VPN软件。3、首次运行谷歌地球,需要点击“文件一登录服务器”,如果软件界面显示黑屏。4、选择“帮助一启动修复工具”。5、先关闭谷歌地球软件,保留“修复Google地球界面”不要关闭。6、选择“恢复默认设置”,窗口不要关闭。7、在次运行谷歌地球软件,点击“文件一一登录服务器”,稍等几秒钟熟悉的地球界面出来后,谷歌地球软件即可正常使用。如果谷歌地球软件无法运行,请在wi

    2026年1月25日
    6
  • 前端面试题ajax_前端性能优化面试题

    前端面试题ajax_前端性能优化面试题AJAX1,Ajax是什么?如何创建一个Ajax?ajax的全称:AsynchronousJavascriptAndXML。异步传输+js+xml。所谓异步,在这里简单地解释就是:向服务器发送请求的时候,我们不必等待结果,而是可以同时做其他的事情,等到有了结果它自己会根据设定进行后续操作,与此同时,页面是不会发生整页刷新的,提高了用户体验(1)创建XMLHttpRequest对象…

    2022年8月27日
    5
  • Vue中watch用法详解

    Vue中watch用法详解基本用法 当 firstName 值变化时 watch 监听到并且执行 div p FullName fullName p p FirstName inputtype text v model firstName inputtype text v model firstName p div newVue el root data firstName Dawei la

    2026年3月16日
    2
  • EXTJS AJAX提交带提示框功能实现

    EXTJS AJAX提交带提示框功能实现

    2021年9月5日
    101
  • 搭建 Drupal 个人网站的图文教程

    搭建 Drupal 个人网站的图文教程操作场景Drupal是使用PHP语言编写的开源内容管理框架(CMF),由内容管理系统(CMS)及PHP开发框架(Framework)共同构成。Drupal具备强大的定制化开发能力,您可使用Drupal作为个人或团体网站开发平台。本文档介绍如何在腾讯云云服务器(CVM)上手动搭建Drupal个人网站。进行手动搭建Drupal个人网站需要熟悉Linux命令,例如Cen…

    2022年6月11日
    36

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号