【数据结构与算法】java有向带权图最短路径算法-Dijkstra算法(通俗易懂)

目录

  • 一、什么是Dijkstra算法
  • 二、算法基本步骤
  • 三、java代码
  • 四、拓展(无向图的Dijkstra算法)

一、什么是Dijkstra算法

Dijkstra算法的核心思想是通过逐步逼近的方式,找出从起点到图中其他所有节点的最短路径。算法的基本步骤如下:

举个例子:

在这里插入图片描述

如图所示的V1-V6六个点及他们的有向权重连线,现在我们假设从V1点出发,画出从顶点V1到其余各点最短路径的过程:

首先,我们将V1拿出来,V1能通向V2和V3,V1到V1的距离我们可以看成0,V1到V2的距离是10,V1到V3的距离是12,V1不能直接到达V4,V5,V6,我们可以看成是无穷大,那么V1的上一个结点是V1,V2和V3的上一个结点也是V1。V4,V5,V6此是没有连接结点记为-1,得到如下表格

在这里插入图片描述

然后根据距离数组{0,10,12,∞,∞,∞},找出数组中距离最小的值,即V2的10,我们将V2拿出来放到数组S中,则数组V中还剩余{V3,V4,V5,V6},现在我们取出了V1,V2;V1到V1和V2的位置还是没有变,取出V2后,V1到V3没有新的通路,所以距离还是12,所以V3的上一个点还是V1;V4和V5是可以根据V2进行跟新的,V4=10+16=26,V5=10=25=35,我们取出了V1,V2,到V6还是没有路线可以走,所以更新之后的表格如下:

在这里插入图片描述

我们距离数组{0,10,12,26,35,∞}中,选取最小值,即12的V3结点加入数组S中,数组V为{V4,V5,V6},现在加入的结点为V1、V2、V3,现在V1到V2的路线多出来V1-V3-V2,但是总长度是15比原本的10要大啊,所以不做变化,V1到V3的距离还是12,V1-V4有两条路(V1-V2-V4)和(V1-V3-V4)跟新后的V1-V3-V4距离是24比原来的26更短,所以替换之,然后V4上一个点的坐标就变成了V3,我们再看一下V1-V5,(V1-V2-V5)和(V1-V3-V2-V5)显然还是原本的35是最短距离;V1-V6的路径是(V1-V3-V6)距离是20,更新表格:

在这里插入图片描述

我们在数组{0,10,12,24,35,20}可以看出在去掉V1、V2、V3之后最小的点是V6的20,所以我们将V6加入到数组S中,V1到V1、V2、V3的距离保持不变;V1到V4的,因为增加了V6,所以多出来一条V1-V3-V6-V4,距离是22,比之前的24小,进行更新,所以V4的上一个结点变成V6;然后V1到V5,多增加路线V1-V3-V6-V5,总体距离变成30,比之前的35要小,更新表格,V5的上一个结点变成V6,跟新后的表格:

在这里插入图片描述

从数组V中取出距离最短的值V4放入数组S中,此时,V1到V1、V2、V3、V4的距离保持不变,V1-V5的距离多了一条V1-V3-V6-V4-V5,路径从29比以前的30要短,更新表格,所以V5的上一个结点的V4,V1-V6保持不变,更新后表格如下:

在这里插入图片描述
将最后一个点V5添加到数组S中,V5没有到其他点的新路径,所以dist[]和path[]数组不变。

如果想要知道V1到V6的距离:

先看path[],V6的上一个结点时V3,V3的上一个结点是V1,所以V1到V6的路径是V1-V3-V6;由dist[]数组得知距离权重是20;

如果想要知道V1到V5的距离:

先看path[],V5的上一个结点时V4,V4的上一个结点时V6,V6的上一个结点时V3,V3的上一个结点是V1,所以V1到V5的路径是V1-V3-V6-V4-V5;由dist[]数组得知距离权重是29;

其他的以此类推;

二、算法基本步骤

  1. 初始化:
  • 创建一个最短路径信息数组shortPath[x][3],每一个一维数组含义为当前结点、该节点到此节点的最短路径、起始节点。
  • 初始化shortPath数组,shortPath[x][0]当前节点编号、shortPath[x][1]最短路径、shortPath[x][2]起始结点编号
  • 初始化优先队列,将起始节点的所有邻接点加入到优先队列中,结点信息使用Ownership类,属性值{time、nodeIndex、weight}。
  • 创建优先队列,优先队列按照结点的权重值优先出队 PriorityQueue。
  • 创建优先队列的比较器OwnershipCustomerComparator类,通过weight大小进行优先出队。
  1. 流程:
  • 优先队列为空则退出
  • 遍历优先队列,将队列中time版本对应的结点信息值写入shortPath中。每次拿到最新路径长度newWeight=matrix[up - 1][ownership.nodeIndex] + shortPath[up - 1][1](起始节点最短路径+起始节点到当前节点一条边的权重),如果当前结点未被初始化则直接将newWeight写入shortPath数组中,如果当前节点已经被写过最短路径,则直接略过当前newWeight即可,这里有一个dtx变量,记录当前优先队列中结点是否被写如果shortPath,用于time(版本)。
  • 遍历优先队列(需要出队),将权重最小的结点出队,将该节点下的所有邻接点拿出做以下操作步骤:
    • 需要是出对节点的邻接点
    • 邻接点在shortPath表中的最短路径未被初始化(还是无穷大),将结点信息写入,最短路径为出对节点权重+出对节点到达该邻接点的权重
    • 查看该邻接点是否出现在优先队列中。在优先队列中则更新shortPath数组以及优先队列中该结点的权重以及起始节点的信息。
    • 优先队列中没有,Math.min(newWeight, shortPath[i][1]),取最优路径写入

三、java代码

代码地址:GitHub

算法代码:

public class Dijkstra {private Queue visited;int[] distance;public Dijkstra(int len) {// TODO Auto-generated constructor stubvisited = new LinkedList();distance = new int[len];}private int getIndex(Queue q, int[] dis) {int k = -1;int min_num = Integer.MAX_VALUE;for (int i = 0; i < dis.length; i++) {if (!q.contains(i)) {if (dis[i] < min_num) {min_num = dis[i];k = i;}}}return k;}public void dijkstra(int[][] weight, Object[] str, int v) {HashMap path;path = new HashMap();for (int i = 0; i < str.length; i++)path.put(i, "");//初始化路径长度数组distancefor (int i = 0; i < str.length; i++) {path.put(i, path.get(i) + "" + str[v]);if (i == v)distance[i] = 0;else if (weight[v][i] != -1) {distance[i] = weight[v][i];path.put(i, path.get(i) + "-->" + str[i]);} elsedistance[i] = Integer.MAX_VALUE;}visited.add(v);while (visited.size() < str.length) {int k = getIndex(visited, distance);//获取未访问点中距离源点最近的点visited.add(k);if (k != -1) {for (int j = 0; j < str.length; j++) {//判断k点能够直接到达的点if (weight[k][j] != -1) {//通过遍历各点,比较是否有比当前更短的路径,有的话,则更新distance,并更新path。if (distance[j] > distance[k] + weight[k][j]) {distance[j] = distance[k] + weight[k][j];path.put(j, path.get(k) + "-->" + str[j]);}}}}}for (int h = 0; h < str.length; h++) {System.out.printf(str[v] + "-->" + str[h] + ":" + distance[h] + " ");if (distance[h] == Integer.MAX_VALUE)System.out.print(str[v] + "-->" + str[h] + "之间没有可通行路径");elseSystem.out.print(str[v] + "-" + str[h] + "之间有最短路径,具体路径为:" + path.get(h).toString());System.out.println();}visited.clear();}
}

测试代码:

public static void main(String[] args) {// TODO Auto-generated method stubint[][] weight = {{0, 10, 12, -1, -1, -1},{-1, 0, -1, 16, 25, -1},{4, 3, 0, 12, -1, 8},{-1, -1, -1, 0, 7, -1},{-1, -1, -1, -1, 0, -1},{-1, -1, -1, 2, -1, 0}};String[] str = {"V1", "V2", "V3", "V4", "V5", "V6"};int len = str.length;Dijkstra dijkstra = new Dijkstra(len);//依次让各点当源点,并调用dijkstra函数for (int i = 0; i < str.length; i++) {dijkstra.dijkstra(weight, str, i);}}

测试结果:
在这里插入图片描述

四、拓展(无向图的Dijkstra算法)

有向图问题解决了,无向图道理和有向图类似,例如如下的无向图,找出V1到其他个点的最短路径

在这里插入图片描述
我们只需要在Test类中定义一个无向图数组

int[][] undirected_weight = {{0,3,-1,7,-1},{3,0,4,2,-1},{-1,4,0,5,4},{7,2,5,0,6},{-1,-1,4,6,0}};
String[] str = {"V1", "V2", "V3", "V4", "V5"};

最后运行结果:

在这里插入图片描述

觉得有用的话还请多多点赞、收藏、评论!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/286996.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

无货源违规又现,现在还能做抖音小店吗?无货源商家该怎么调整?

大家好&#xff0c;我是电商花花。 最近好像又有很多人的店铺被查无货源违规&#xff0c;店铺还被扣12分&#xff0c;也申诉不了。 如果想要长期的做下去&#xff0c;就不要秀那些花里胡哨的操作&#xff0c;也不要为了短暂的自然流量而进行违规操作&#xff0c;为什么你的店…

【Java程序设计】【C00351】基于Springboot的疫情居家办公系统(有论文)

基于Springboot的疫情居家办公系统&#xff08;有论文&#xff09; 项目简介项目获取开发环境项目技术运行截图 项目简介 项目获取 &#x1f345;文末点击卡片获取源码&#x1f345; 开发环境 运行环境&#xff1a;推荐jdk1.8&#xff1b; 开发工具&#xff1a;eclipse以及i…

后端常见面经之MySQL

MySQL字段类型 数值类型 整型经常被用到&#xff0c;比如 tinyint、int、bigint 。默认是有符号的&#xff0c;若只需存储无符号值&#xff0c;可增加 unsigned 属性。 int(M)中的 M 代表最大显示宽度&#xff0c;并不是说 int(1) 就不能存储数值10了&#xff0c;不管设定了显…

docker容器下部署hbase并在springboot中通过jdbc连接

我在windows的docker中部署了一个hbase服务&#xff0c;然后用springboot连接到此服务并访问数据。 详情可参考项目中的README.md。项目中提供了用于构建镜像的dockerfile&#xff0c;以及测试代码。 项目连接&#xff1a; https://gitee.com/forgot940629/hbase_phoenix_sprin…

java常用IO流功能——字符流和缓冲流概述

前言&#xff1a; 整理下学习笔记&#xff0c;打好基础&#xff0c;daydayup! 之前说了下了IO流的概念&#xff0c;并整理了字节流&#xff0c;有需要的可以看这篇 java常用应用程序编程接口&#xff08;API&#xff09;——IO流概述及字节流的使用 字符流 FileReader(文件字…

文件一键加水印的软件下载

文件一键加水印的软件通常具有强大的功能特点。 首先&#xff0c;它们支持多种文件格式&#xff0c;无论是常见的图片、文档&#xff0c;还是视频、音频文件&#xff0c;都能轻松应对。这极大地提高了软件的适用性和实用性。 其次&#xff0c;这些软件通常提供多种水印样式和…

【动态规划】【卡特兰数】Leetcode 96. 不同的二叉搜索树

【动态规划】【卡特兰数】Leetcode 96. 不同的二叉搜索树 动态规划卡特兰数 ---------------&#x1f388;&#x1f388;96. 不同的二叉搜索树 题目链接&#x1f388;&#x1f388;------------------- 动态规划 &#x1f612;: 我的代码实现> 动规五部曲 ✒️确定dp数组…

一款不错的开源的 Linux 服务器运维管理面板:1Panel

适用于非运维人员的环境搭建、部署、监控等 一、1Panel 是一个现代化、开源的 Linux 服务器运维管理面板。1Panel 的功能和优势包括&#xff1a; 快速建站&#xff1a;深度集成 Wordpress 和 Halo&#xff0c;域名绑定、SSL 证书配置等一键搞定&#xff1b; 高效管理&#xf…

QT作业day3

1、使用手动连接&#xff0c;将登录框中的取消按钮使用qt4版本的连接到自定义的槽函数中&#xff0c;在自定义的槽函数中调用关闭函数 将登录按钮使用qt5版本的连接到自定义的槽函数中&#xff0c;在槽函数中判断ui界面上输入的账号是否为"admin"&#xff0c;密码是…

常见端口及对应服务

6379 redis未授权 7001、7002 weblogic默认弱口令、反序列化 9200、9300 elasticsearch 参考乌云&#xff1a;多玩某服务器ElasticSearch命令执行漏洞 11211 memcache未授权访问 50000 SAP命令执行 50070、50030 hadoop默认端口未授权访问

创建数组的时候,数组大小是确定数值和变量的不同情况

概要&#xff1a; 1、创将数组的时候&#xff0c;如果数组大小是确定数值 &#xff08;1&#xff09;数组所有元素默认是0 &#xff08;2&#xff09;可以通过大括号对元素进行赋值 int arr[3]{1,2,3}; int arr[10]{1}; //只将第一个元素赋值为1,其他元素依然是0 2、…

【机器学习】无监督学习算法之:K均值聚类

K均值聚类 1、引言2、K均值聚类2.1 定义2.2 原理2.3 实现方式2.4 算法公式2.4.1 距离计算公式2.4.1 中心点计算公式 2.5 代码示例 3、总结 1、引言 小屌丝&#xff1a;鱼哥&#xff0c; K均值聚类 我不懂&#xff0c;能不能给我讲一讲&#xff1f; 小鱼&#xff1a;行&#xf…

距离AI PC起飞,还差了点什么?

作者 | 张未 来源 | 洞见新研社 PC行业也没有逃过万物皆可AI的真香定律。 英伟达在前喊出AI PC的口号后&#xff0c;一众PC厂商纷纷加码这一最新概念&#xff0c;有关AI PC的讨论点燃了PC市场。 最直观的变化就是&#xff0c;全球PC市场终于止住了颓势&#xff0c;打破了七连…

雪里温柔,水边明秀,不及Java 抽象类 和 Object类

本篇会加入个人的所谓‘鱼式疯言’ ❤️❤️❤️鱼式疯言:❤️❤️❤️此疯言非彼疯言 而是理解过并总结出来通俗易懂的大白话, 小编会尽可能的在每个概念后插入鱼式疯言,帮助大家理解的. &#x1f92d;&#x1f92d;&#x1f92d;可能说的不是那么严谨.但小编初心是能让更多人…

【测试开发学习历程】认识Python + 安装Python

目录 1 认识 Python 1.1 Python 的起源 1.2 Python的组成 1.2.1 解释器 1.1.2 Python 的设计目标 1.1.3 Python 的设计哲学 1.2 为什么选择 Python 测试人员选择Python的理由 1.3 Python 特点 面向对象的思维方式 1.4 Python 的优缺点 1.4.1 优点 1.4.2 缺点 3. 安…

哈希冲突解决的几种方式

目录 哈希冲突 哈希冲突-避免方式1-哈希函数的设计 1. 直接定制法--(常用) 2. 除留余数法--(常用) 3. 平方取中法--(了解) 哈希冲突-避免方式2-负载因子调节 哈希冲突-解决方式1-闭散列 1.线性探测 2.二次探测 哈希冲突-解决方式2-开散列(哈希桶) 哈希冲突 在上文中…

es bulk批量操作简单实例

&#xff08;1&#xff09;定义 bulk允许在单个步骤中进行多次create、index、update或delete请求。 bulk与其他的请求体格式稍有不同&#xff0c;如下所示&#xff1a; { action: { metadata }}\n { request body }\n { action: { metadata }}\n { request body …

智慧医疗包括哪些方面?智慧医疗发展前景如何?

近年来&#xff0c;随着云计算、物联网&#xff08;internet of things&#xff0c;IOT&#xff09;、移动互联网、大数据、人工智能&#xff08;artificial intelligence&#xff0c;AI&#xff09;、5G网络、区块链等新一代信息技术的逐步成熟和广泛应用&#xff0c;信息化已…

linux系统编程 socket part2

报式套接字 1.动态报式套接字2.报式套接字的广播3.报式套接字的多播4.UDP协议分析4.1.丢包原因4.2.停等式流量控制 接linux系统编程 socket part1 1.动态报式套接字 在之前的例子上&#xff0c;发送的结构体中的名字由定长改变长。可以用变长结构体。 变长结构体是由gcc扩展的…

判断python字典中key是否存在的两种方法

今天来说一下如何判断字典中是否存在某个key&#xff0c;一般有两种通用做法&#xff0c;下面为大家来分别讲解一下&#xff1a;第一种方法&#xff1a;使用自带函数实现。 在python的字典的属性方法里面有一个has_key()方法&#xff0c;这个方法使用起来非常简单。 例&#xf…