【R】Dijkstra算法求最短路径

embedded/2025/2/8 20:25:37/

使用R语言实现Dijkstra算法求最短路径

求点2、3、4、5、6、7到点1的最短距离和路径

1.设置data,存放有向图信息

data中每个点所在的行序号为起始点序号,列为终点序号。

比如:值4的坐标为(1,2)即点1到点2距离为4;值8的坐标为(6,7)即点6到点7距离为8;INF表示x->y不通。

代码 

m <- Inf
num <- 7
data <- matrix(c(m, 4, 6, 6, m, m, m,m, m, 1, m, 7, m, m,m, m, m, m, 6, 4, m,m, m, 2, m, m, 5, m,m, m, m, m, m, m, 6,m, m, m, m, 1, m, 8,m, m, m, m, m, m, m
), nrow = num, byrow = TRUE)

2.辅助参数设置 

dist:每个点到初始点的距离,设置为INF,之后根据data中距离进行更新,最后可以得到每个点到初始点的最短距离
path:初始点到每个点的最短路线中,每个点前面的路径;比如点1->点3的最短路径为点1->点2->点3;那么最后path[3]中存的即是节点2
mark:起始值为0,代表还没有进行更新(找到到起始点的最短距离)
后续使用双重for循环,在最初点开始一轮更新后,以每个点为起始点继续更新(以该点为起始更新完后,该点前面的最小距离已经确定,不再参与后续更新,其mark被设置为1),这个过程需要遍历每个点

代码 

dist <- rep(m,num)
path <- rep(-1,num)
mark <- rep(0,num)

此处参考资料:Dijkstra算法求最短路径_哔哩哔哩_bilibili

3.Min():更新过程中分离出来的便捷函数

在全部的节点中寻找mark[i] == 0,即没访问过的点中有最小值的点,返回w,作为下一轮内循环的k,开始更新联通的点

代码 

Min <- function(last){w = 1while (mark[w] != 0){w <- w+1}for (i in 1:last){if (mark[i] == 0 && dist[i] < dist[w]){w = i}}return(w)
}

4.Dijkstra算法主要过程 

4.1.起点初始化

4.1初始点为点1,更新其三个参数

代码 

k = 1
dist[k] = 0
path[k] = -1
mark[k] = 1

4.2.内外循环  

内循环:如果dist[i] > dist[k] + data[k,i]成立,说明k点到i点是通的(不是INF),即根据点k更新点i(和点k联通的多个点)
比如从点1开始(初始k的距离为0:dist[k] = 0)遍历全部点,更新联通的点2,3,4,结束第一次内循环
外循环:第一次内循环结束后,使用函数Min()找和点1联通的点中dist最小的点,标记为mark[k] = 1,结束第一次外循环,第二轮内循环的k为和点1联通的这个最近点
因为基于点k更新的联通点的距离是基于dist[k]的,此时被函数Min()找出,标记为mark[k] = 1,不再参加后续更新
注意k的顺序不是按序号来的,取决于data,比如点2,3,4和1联通,第二次从2开始更新和2联通的3和5,点3就还可以更新,接着从点3开始更新5和6,但是接着从4开始,只取决于距离
接着遍历全部的点,找到目前距离最短的,没有完成更新的点,标记结束后作为下一次更新的起始点

代码 

for(x in 2:num){ # 遍历全部的点for (i in 2:num){ # 以点1开始更新联通点,其实因为有if (mark[i] == 0),所以2:num还是1:num无所谓if (mark[i] == 0){if (dist[i] > dist[k] + data[k,i]){dist[i] = dist[k] + data[k,i]path[i] = k}}}k = Min(num) # 从联通点中距离最短的点开始继续更新该点的联通点 mark[k] = 1 # 比如,点1的联通点中最小点为2,那么下一轮从2开始,同时点2不再更新,点3、4还有更新的机会
}

5.可视化 

代码 

## 打印终点到终点的最短距离----
cat("Shortest distance to node",num,":", dist[num], "\n")## 打印每个节点的最短路径和距离----
for (u in 1:num){cat(sprintf("Shortest Path to node %d: %d -> %d\n", u, path[u], u))cat(sprintf("Minimum Distance to node %d: %d\n", u, dist[u]))
}

完整代码

# 设置data,存放有向图信息----
## data中每个点所在的行序号为起始点序号,列为终点序号----
### 比如:值4的坐标为(1,2)即点1到点2距离为4;值8的坐标为(6,7)即点6到点7距离为8;INF表示x->y不通----
m <- Inf
num <- 7
data <- matrix(c(m, 4, 6, 6, m, m, m,m, m, 1, m, 7, m, m,m, m, m, m, 6, 4, m,m, m, 2, m, m, 5, m,m, m, m, m, m, m, 6,m, m, m, m, 1, m, 8,m, m, m, m, m, m, m
), nrow = num, byrow = TRUE)# 辅助参数----
## dist:每个点到初始点的距离,设置为INF,之后根据data中距离进行更新,最后可以得到每个点到初始点的最短距离----
## path:初始点到每个点的最短路线中,每个点前面的路径;比如点1->点3的最短路径为点1->点2->点3;那么最后path[3]中存的即是节点2----
## mark:起始值为0,代表还没有进行更新(找到到起始点的最短距离)----
## 后续使用双重for循环,在最初点开始一轮更新后,以每个点为起始点继续更新(以该点为起始更新完后,该点前面的最小距离已经确定,不再参与后续更新,其mark被设置为1),这个过程需要遍历每个点----
dist <- rep(m,num)
path <- rep(-1,num)
mark <- rep(0,num)
### 此处参考资料:https://www.bilibili.com/video/BV11P4y1a7X9/?spm_id_from=333.999.0.0&vd_source=709bfcb2343a93fce7f20d52e9a6c8cf# 更新过程中分离出来的便捷函数----
## 在全部的节点中寻找mark[i] == 0,即没访问过的点中有最小值的点,返回w,作为下一轮内循环的k,开始更新联通的点----
Min <- function(last){w = 1while (mark[w] != 0){w <- w+1}for (i in 1:last){if (mark[i] == 0 && dist[i] < dist[w]){w = i}}return(w)
}# Dijkstra算法主要过程----
## 初始点为点1,更新其三个参数
k = 1
dist[k] = 0
path[k] = -1
mark[k] = 1
## 内循环:如果dist[i] > dist[k] + data[k,i]成立,说明k点到i点是通的(不是INF),即根据点k更新点i(和点k联通的多个点)
### 比如从点1开始(初始k的距离为0:dist[k] = 0)遍历全部点,更新联通的点2,3,4,结束第一次内循环
## 外循环:第一次内循环结束后,使用函数Min()找和点1联通的点中dist最小的点,标记为mark[k] = 1,结束第一次外循环,第二轮内循环的k为和点1联通的这个最近点
### 因为基于点k更新的联通点的距离是基于dist[k]的,此时被函数Min()找出,标记为mark[k] = 1,不再参加后续更新
### 注意k的顺序不是按序号来的,取决于data,比如点2,3,4和1联通,第二次从2开始更新和2联通的3和5,点3就还可以更新,接着从点3开始更新5和6,但是接着从4开始,只取决于距离
### 接着遍历全部的点,找到目前距离最短的,没有完成更新的点,标记结束后作为下一次更新的起始点
for(x in 2:num){ # 遍历全部的点for (i in 2:num){ # 以点1开始更新联通点,其实因为有if (mark[i] == 0),所以2:num还是1:num无所谓if (mark[i] == 0){if (dist[i] > dist[k] + data[k,i]){dist[i] = dist[k] + data[k,i]path[i] = k}}}k = Min(num) # 从联通点中距离最短的点开始继续更新该点的联通点 mark[k] = 1 # 比如,点1的联通点中最小点为2,那么下一轮从2开始,同时点2不再更新,点3、4还有更新的机会
}# 可视化----
## 打印终点到终点的最短距离----
cat("Shortest distance to node",num,":", dist[num], "\n")## 打印每个节点的最短路径和距离----
for (u in 1:num){cat(sprintf("Shortest Path to node %d: %d -> %d\n", u, path[u], u))cat(sprintf("Minimum Distance to node %d: %d\n", u, dist[u]))
}

http://www.ppmy.cn/embedded/160614.html

相关文章

RabbitMQ 与 Kafka 的核心区别,如何选择合适的消息中间件?

在现代分布式系统中&#xff0c;消息队列&#xff08;Message Queue&#xff0c;MQ&#xff09;扮演着重要角色&#xff0c;能够解耦服务、提高系统伸缩性、增强可靠性。目前&#xff0c;RabbitMQ 和 Kafka 是两款最常见的消息中间件&#xff0c;它们虽然都能实现消息传输&…

elasticsearch(ES)简介及安装-----笔记

elasticsearch简介 ES是一款非常强大的开源搜索引擎&#xff0c;可以帮助我们从海量数据中快速找到需要的内容。 ES结合kibana、Logstash、Beats&#xff0c;也就是elastic stack(ELK)。被广泛应用在日志数据分析、实时监控等领域。 elasticsearch是elastic stack的核心&…

【Elasticsearch】geotile grid聚合

geotile_grid聚合是 Elasticsearch 中一种用于处理地理数据的多桶聚合方式&#xff0c;它将geo_point和geo_shape类型的值分组到表示网格的桶中。以下是关于geotile_grid聚合的详细说明&#xff1a; 基本概念 • 网格划分&#xff1a;geotile_grid聚合将地理数据划分为一个稀疏…

【mysql】数据库字段设计原则

本文将分享17个关键字段设计原则&#xff0c;这些经验可规避80%的数据库设计缺陷&#xff0c;涵盖性能、扩展性、可维护性等核心维度&#xff0c;附具体场景示例&#xff1a; 一、数据类型选择&#xff1a;避免“隐形成本杀手” 1. 整数类型精确匹配 坑&#xff1a;滥用BIGIN…

Llama最新开源大模型Llama3.1

Meta公司于2024年7月23日发布了最新的开源大模型Llama 3.1&#xff0c;这是其在大语言模型领域的重要进展。以下是关于Llama 3.1的详细介绍&#xff1a; 参数规模与训练数据 Llama 3.1拥有4050亿&#xff08;405B&#xff09;参数&#xff0c;是目前开源领域中参数规模最大的…

为多个GitHub账户配置SSH密钥

背景 当需要同时使用多个GitHub账户&#xff08;例如工作和个人账户&#xff09;时&#xff0c;默认的SSH配置可能导致冲突。本文介绍如何通过生成不同的SSH密钥对并配置SSH客户端来管理多个账户。 操作步骤 生成SSH密钥对 为每个GitHub账户生成独立的密钥对&#xff0c;并指…

DeePseek结合PS!批量处理图片的方法教程

​ ​ 今天我们来聊聊如何利用deepseek和Photoshop&#xff08;PS&#xff09;实现图片的批量处理。 传统上&#xff0c;批量修改图片尺寸、分辨率等任务往往需要编写脚本或手动处理&#xff0c;而现在有了AI的辅助&#xff0c;我们可以轻松生成PS脚本&#xff0c;实现自动化处…

电子电器架构 --- 电子电气架构设计要求与发展方向

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 简单,单纯,喜欢独处,独来独往,不易合同频过着接地气的生活,除了生存温饱问题之外,没有什么过多的欲望,表面看起来很高冷,内心热情,如果你身…