[算法]-最短路径算法总结

article/2025/8/25 2:59:04

Dijkstra最短路径算法

按路径长度的递增次序,逐步产生最短路径的贪心算法

基本思想:首先求出长度最短的一条最短路径,再参照它求出长度次短的一条最短路径,依次类推,直到从顶点v 到其它各顶点的最短路径全部求出为止。

时间复杂度为O(n2)

算法流程:
在这里插入图片描述

  1. 首先选定源点1,建立邻接矩阵C[5] [5],初始化三个数组分别为D[n],P[n],S[n],分别用来存储从源点到对应点的最短距离和最短路径上到该序列节点的前序节点,以及是否已经找到最短路径。
  2. 开始遍历源点的出边,将邻接矩阵的第一行放入数组D,找出距离最小的节点序号2,将数组P中的P[2]置为1,因为2的前序节点为1。
  3. 以上一步骤找到的节点为起点,继续遍历其邻接点(此处为2),若C[1][2]+C[2][m]<D[m] 则将其替换进数组D中,并将数组P中的P[m]置为2,因为m最短路径上的前序节点为2,节点2的邻接点全部遍历完成后,再从数组D中找出值最小且未被选中过的节点
  4. 重复以上步骤3,直到所有点都已经加入到数组S中。
    在这里插入图片描述

程序实现:

public class Dijikstra extends Strategy {// open表Map<Integer, Node> open = new HashMap<>();// close表Map<Integer, Node> close = new HashMap<>();// 动作列表int[][] motion = get_motion();@Overridepublic List<Location> nextstep(Location start, Location work) {Node startnode = new Node(start.getX(), start.getY(), 0, -1);Node worknode = new Node(work.getX(), work.getY(), 0, -1);// 将起点放入open表open.put(cal_grid_index(startnode), startnode);// 向起点的上下左右四个方向扩展while (true) {// 与Astar算法不同 A*算法含有启发式函数 可以保证更快找到目标点// 但A*算法有可能永远无法找到目标点// Dijikstra算法虽然较慢 会遍历全部方向的点 但一定可以找到一条到目标点的路径// 找到map中节点cost最小的元素int temp_cost = Integer.MAX_VALUE;int current_id = 0;Node current = null;for (Map.Entry<Integer, Node> entry : open.entrySet()) {int current_cost =entry.getValue().cost + cal_Manhattan_distance(entry.getValue(), worknode);if (current_cost < temp_cost) {current_id = entry.getKey();temp_cost = current_cost;current = entry.getValue();}}// 判断是否找到目标点if (current.x == work.getX() && current.y == work.getY()) {System.out.println("找到目标点!");worknode.pind = current.pind;worknode.cost = current.cost;break;}// 移除open表中的current节点for (Iterator<Integer> iterator = open.keySet().iterator(); iterator.hasNext();) {Integer key = iterator.next();if (key == current_id) {iterator.remove();}}// 将current节点放入close表中close.put(current_id, current);for (int i = 0; i < motion.length; i++) {Node node = new Node(current.x + motion[i][0], current.y + motion[i][1],current.cost + motion[i][2], current_id);int node_id = cal_grid_index(node);// 该节点不能被扩展if (!verify_node(node)) {continue;}// 该节点已被确认过if (close.containsKey(node_id)) {continue;}// 将新扩展节点放入open表中if (!open.containsKey(node_id)) {open.put(node_id, node);} else {// 更新open表if (open.get(node_id).cost > node.cost) {open.put(node_id, node);}}}}open.clear();return cal_final_path(worknode, close);}/*** 根据终点回溯计算最短路径* * @param worknode* @param close* @return*/private List<Location> cal_final_path(Node worknode, Map<Integer, Node> close) {List<Location> ans = new ArrayList<>();int pind = worknode.pind;int i = 0;while (pind != -1) {// System.out.println(i);Node node = close.get(pind);Location location = new Location(node.x, node.y);ans.add(location);pind = node.pind;}return ans;}/*** 计算曼哈顿距离* * @param now* @param end* @return*/private int cal_Manhattan_distance(Node now, Node end) {return Math.abs(now.x - end.x) + Math.abs(now.y - end.y);}/*** 判断节点是否合理 1. 是否超出范围 2. 是否遇到货柜* * @param node* @return*/private boolean verify_node(Node node) {Container[][] map = ContainerMap.getMap();if (node.x < 0 || node.x >= ContainerMap.N || node.y < 0 || node.y >= ContainerMap.N) {return false;}if (map[node.x][node.y] != null) {return false;}return true;}/*** 计算栅格地图中的线性坐标* * @param node* @return*/private int cal_grid_index(Node node) {return node.y * ContainerMap.N + node.x;}/*** 定义动作及其代价 motion = {dx, dy, cost}* * @return*/private int[][] get_motion() {int[][] motion = {{1, 0, 1}, {0, 1, 1}, {-1, 0, 1}, {0, -1, 1}};return motion;}
}

A*算法

利用A*算法在拓扑地图上实现的路径规划:
关键:代价函数

  • 对于任意节点n
    • g(n)=从树根到n的代价
    • h*(n)=从n到目标节点的优化路径的代价
    • f*(n)=g(n) + h*(n)是节点n的代价
  • 估计h*(n)
    • 使用任何方法去估计h*(n), 用h(n)表示h*(n)的估计
    • h(n) <= h*(n)总为真
    • f(n)=g(n)+h(n) <= g(n)+h*(n)=f*(n)定义为n的代价

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

利用A*算法在栅格地图上实现的路径规划:
A*算法的核心思想是通过设置代价函数对未遍历点进行筛选,使搜索方向更偏向目标点,缩小搜索范围,进而提升搜索速度。公式f(n)=g(n)+h(n)是路径成本表达式。其中,f(n)为起点P—遍历中间点n—目标点g的路径总成本;g(n)为起点P—遍历中间点n之间的实际路径成本;h(n)为遍历中间点n—目标点Q的预估路径成本。寻找到最优解的条件是f(n)尽可能接近起点P到目标点Q的真实路径成本,此时搜寻到的点都是最短路径的过程点。由于遍历中间点n的g(n)值己知,因此只能提升h(n)的估计准确性,根据AGV在作业环境中只有“上下左右”四个行走方向,所以选取 h(n) = |Xn - XQ|+|Yn - YQ| (即曼哈顿距离),
算法流程描述如下:
(1)新建两个列表用作路径数据点存储:open list和close list,open list初始默认包含路径起点;
(2)计算与起点相邻的节点的f(n),按数值大小进行排序,f(n)最小的节点m转移到close list作为当前最优路径过程点,其余点留在open list中做备选点。
(3)若m为目标点,则停止搜索,并输出最优路径结果,否则继续;
(4)计算遍历中间栅格点m的所有相邻栅格点的f(n),若相邻节点n未加入open list,及时添加,并令g(n) = g(m) + d(m,n),此时n的父节点为m;若相邻点n在open list中,当g(n) > g(m) + d(m,n)时,更新g(n) = g(m) + d(m,n),设置n的父节点为m;
(5)遍历完所有备选点后重复步骤(2)-(4),直到目标点Q出现在open list中,搜索结束,按进入closed list的先后顺序输出中的节点栅格序列,该序列即为最优路径。

程序实现:

public class Astar extends Strategy {// open表Map<Integer, Node> open = new HashMap<>();// close表Map<Integer, Node> close = new HashMap<>();// 动作列表int[][] motion = get_motion();@Overridepublic List<Location> nextstep(Location start, Location work) {Node startnode = new Node(start.getX(), start.getY(), 0, -1);Node worknode = new Node(work.getX(), work.getY(), 0, -1);// 将起点放入open表open.put(cal_grid_index(startnode), startnode);// 向起点的上下左右四个方向扩展while (true) {if (open.size() == 0) {System.out.println("open表空!查找失败!");break;}// 找到map中节点cost最小的元素int temp_cost = Integer.MAX_VALUE;int current_id = 0;Node current = null;for (Map.Entry<Integer, Node> entry : open.entrySet()) {int current_cost =entry.getValue().cost + cal_Manhattan_distance(entry.getValue(), worknode);if (current_cost < temp_cost) {current_id = entry.getKey();temp_cost = current_cost;current = entry.getValue();}}// 判断是否找到目标点if (current.x == work.getX() && current.y == work.getY()) {System.out.println("找到目标点!");worknode.pind = current.pind;worknode.cost = current.cost;break;}// 移除open表中的current节点for (Iterator<Integer> iterator = open.keySet().iterator(); iterator.hasNext();) {Integer key = iterator.next();if (key == current_id) {iterator.remove();}}// 将current节点放入close表中close.put(current_id, current);for (int i = 0; i < motion.length; i++) {Node node = new Node(current.x + motion[i][0], current.y + motion[i][1],current.cost + motion[i][2], current_id);int node_id = cal_grid_index(node);// 该节点不能被扩展则查找下一个节点if (!verify_node(node)) {continue;}// 该节点已被确认过则查找下一个节点if (close.containsKey(node_id)) {continue;}// 将新扩展节点放入open表中if (!open.containsKey(node_id)) {open.put(node_id, node);} else {// 更新节点的代价if (open.get(node_id).cost > node.cost) {open.put(node_id, node);}}}}open.clear();return cal_final_path(worknode, close);}/*** 根据终点回溯计算最短路径* * @param worknode* @param close* @return*/private List<Location> cal_final_path(Node worknode, Map<Integer, Node> close) {List<Location> ans = new ArrayList<>();int pind = worknode.pind;int i = 0;while (pind != -1) {// System.out.println(i);Node node = close.get(pind);Location location = new Location(node.x, node.y);ans.add(location);pind = node.pind;}return ans;}/*** 计算曼哈顿距离* * @param now* @param end* @return*/private int cal_Manhattan_distance(Node now, Node end) {return Math.abs(now.x - end.x) + Math.abs(now.y - end.y);}/*** 判断节点是否合理 1. 是否超出范围 2. 是否遇到货柜* * @param node* @return*/private boolean verify_node(Node node) {Container[][] map = ContainerMap.getMap();if (node.x < 0 || node.x >= ContainerMap.N || node.y < 0 || node.y >= ContainerMap.N) {return false;}if (map[node.x][node.y] != null) {return false;}return true;}/*** 计算栅格地图中的线性坐标* * @param node* @return*/private int cal_grid_index(Node node) {return node.y * ContainerMap.N + node.x;}/*** 定义动作及其代价 motion = {dx, dy, cost}* * @return*/private int[][] get_motion() {int[][] motion = {{1, 0, 1}, {0, 1, 1}, {-1, 0, 1}, {0, -1, 1}};return motion;}
}

http://chatgpt.dhexx.cn/article/9dL1pIin.shtml

相关文章

c++实现最短路径算法(Dijkstra算法)

0简介 Dijkstra算法是一种计算某一顶点到其余顶点最短路径的算法。它采用了贪心的策略&#xff0c;实现时使用了广度优先。这个算法常学常忘&#xff0c;因此写篇博客彻底解决它。 1计算步骤介绍 Dijkstra的计算步骤不难&#xff0c;如下 (1)设置一个集合表示已经标记过的节…

最短路径算法——dijkstra

dijkstra 前提&#xff1a;在一张图里&#xff0c;不能有权值为负数的边。 给你一个出发点&#xff0c;求出发点到所有结点的最短距离是多少&#xff1f;如果无法到达某个点&#xff0c;则到这个点的距离是正无穷&#xff08;一般出现在有向图里面&#xff09;。 举个&#…

最短路径算法详解

前言&#xff1a; 最短路径算法&#xff1a;用于计算一个节点到其他所有节点的最短路径。主要特点是以起始点为中心向外层层扩展&#xff0c;直到扩展到终点为止。 例子&#xff1a; 暑期&#xff0c;你想要出门去旅游&#xff0c;但是在你出发之前&#xff0c;你想知道任意…

基于java最短路径算法的简单实现

一、我们先画一个可达路径图表&#xff1a; -1表示不可直达&#xff0c;0表示自己&#xff0c;其他整数表示长度或者权重 ABCDA012-1B-1015C-1-101D-1-1-10 用图形可表示为&#xff1a; 二、算法思路 1、先列出所有保存所有节点到可达节点的路径二维表 2、逐步寻找从起始节点到…

最短路径算法及应用

乘汽车旅行的人总希望找出到目的地的尽可能的短的行程。如果有一张地图并在图上标出每对十字路口之间的距离&#xff0c;如何找出这一最短行程&#xff1f;  一种可能的方法就是枚举出所有路径&#xff0c;并计算出每条路径的长度&#xff0c;然后选择最短的一条。那么我们很…

关于最短路径算法的理解

“最短路径算法:Dijkstra算法&#xff0c;Bellman-Ford算法&#xff0c;Floyd算法和SPFA算法等。​从某顶点出发&#xff0c;沿图的边到达另一顶点所经过的路径中&#xff0c;各边上权值之和最小的一条路径叫做最短路径。” 我们解决最短路径问题&#xff0c;常用的是Dijkstra…

最短路径算法-----Dijkstra迪杰斯特拉算法

最近巩固一下算法&#xff0c;提高自己内力&#xff0c;网上看到查看到这篇介绍很详细的《Dijkstra迪杰斯特拉算法》&#xff0c;在这里转载记录一下。 1 前言 本章介绍迪杰斯特拉算法。和以往一样&#xff0c;本文会先对迪杰斯特拉算法的理论论知识进行介绍&#xff0c;然后给…

Dijkstra 最短路径算法 Python 实现

原文链接 问题描述 使用 Dijkstra 算法求图中的任意顶点到其它顶点的最短路径&#xff08;求出需要经过那些点以及最短距离&#xff09;。 以下图为例&#xff1a; 算法思想 可以使用二维数组来存储顶点之间边的关系 首先需要用一个一维数组 dis 来存储 初始顶点到其余各个顶…

神经网络最短路径算法,最短路径算法的原理

节约里程法求解最短路问题 你只要记住2点之间直线最短。节约里程法是用来解决运输车辆数目不确定的问题的最有名的启发式算法。1、节约里程法优化过程分为并行方式和串行方式两种。 核心思想是依次将运输问题中的两个回路合并为一个回路&#xff0c;每次使合并后的总运输距离…

最短路径算法及Python实现

最短路径问题 在图论中&#xff0c;最短路径问题是指在一个有向或无向的加权图中找到从一个起点到一个终点的最短路径。这个问题是计算机科学中的一个经典问题&#xff0c;也是许多实际问题的基础&#xff0c;例如路线规划、通信网络设计和交通流量优化等。在这个问题中&#…

图论:图的四种最短路径算法

目录&#xff1a; 1.DFS&#xff08;单源最短路径算法&#xff09; 例题1&#xff1a; DFS题目分析&#xff1a; 代码DFS&#xff1a; 2.Floyed&#xff08;时间复杂度On^3&#xff09; 1.应用场景&#xff1a; 2.解析算法&#xff1a; 核心代码1&#xff1a; 我的笔…

图的五种最短路径算法

本文总结了图的几种最短路径算法的实现:深度或广度优先搜索算法,费罗伊德算法,迪杰斯特拉算法,Bellman-Ford 算法。 1)深度或广度优先搜索算法(解决单源最短路径) 从起点开始访问所有深度遍历路径或广度优先路径,则到达终点节点的路径有多条,取其中路径权值最短的一…

最短路径算法——Dijkstra算法——python3实现

本文参考来自数据结构与算法分析 java语言描述。 文章目录 问题描述问题分析实现过程如何使用数据变化表代码实现优先队列中的堆排序使用set代替优先队列得到最短路径 负权边算法改进&#xff08;若为无圈图&#xff09; 问题描述 现有一个有向赋权图。如下图所示&#xff1a;…

最短路径算法的编程与实现 C语言

一 、目的&#xff1a; 1.掌握最短路径算法的基本原理及编程实现&#xff1b; 二 、环境&#xff1a; operating system version&#xff1a;Win11 CPU instruction set: x64 Integrated Development Environment&#xff1a;Viusal Studio 2022 三 、内容&#xff1a; 1…

图的四种最短路径算法

本文总结了图的几种最短路径算法的实现&#xff1a;深度或广度优先搜索算法&#xff0c;弗洛伊德算法&#xff0c;迪杰斯特拉算法&#xff0c;Bellman-Ford算法 1&#xff09;&#xff0c;深度或广度优先搜索算法&#xff08;解决单源最短路径&#xff09; 从起始结点开始访问所…

算法之几个常见的经典最短路径算法

目录 1. Dijkstra算法2. Floyd算法3. Bellman-Ford 算法 1. Dijkstra算法 是解单源最短路径问题的贪心算法。 有一向带权图 G (V, E)&#xff0c;包含右n个顶点&#xff0c;其中每条边的权是非负实数&#xff0c;定义数组 dist 为原点到G中各个顶点的距离&#xff0c;初始化为…

最短路径的四种算法

最短路径四种算法 1234FloydDijkstraBellman-Ford队列优化的Bellman-Ford 一&#xff0c;只有四行的算法——Floyd-Warshall 假设求顶点 V i Vi Vi到 V j Vj Vj的最短路径。弗洛伊德算法依次找从 V i Vi Vi到 V j Vj Vj&#xff0c;中间经过结点序号不大于 0 0 0的最短路径&…

最短路径算法

1.最短路径算法分为单源最短路径算法和多源最短路径算法 &#xff08;a&#xff09;单源最短路径算法&#xff0c;可以计算出从起点到任意一个起点的最短路径。 例如&#xff1a;Dijkstra算法 &#xff0c;SPFA算法 &#xff08;b&#xff09;多源最短路径算法&#xff0c;可…

哈夫曼树及其应用

1、哈夫曼树的基本概念 ---- 哈夫曼&#xff08;Huffman&#xff09;树又称作最优二叉树&#xff0c;它是n个带权叶子结点构成的所有二叉树中&#xff0c;带权路径长度最小的二叉树。 ---- “路径”就是从树中的一个结点到另一个结点之间的分支构成的部分&#xff0c;而分支…

哈夫曼树的C语言实现

什么是哈夫曼树 当用 n 个结点&#xff08;都做叶子结点且都有各自的权值&#xff09;试图构建一棵树时&#xff0c;如果构建的这棵树的带权路径长度最小&#xff0c;称这棵树为“最优二叉树”&#xff0c;有时也叫“赫夫曼树”或者“哈夫曼树”。 在构建哈弗曼树时&#xff0…