图:两点之间的最短距离

文章出处:极客时间《数据结构和算法之美》-作者:王争。该系列文章是本人的学习笔记。

1 问题阐述

像 Google 地图、百度地图、高德地图这样的地图软件,如果想从家开车到公司,你只需要输入起始、结束地址,地图就会给你规划一条最优出行路线。这里的最优,有很多种定义,比如最短路线、最少用时路线、最少红绿灯路线等等。我们先解决最简单的,最短路线。

2 算法解析

解决软件开发中的实际问题,最重要的一点就是建模,也就是将复杂的场景抽象成具体的数据结构。我们把每个岔路口看作一个顶点,岔路口与岔路口之间的路看作一条边,路的长度就是边的权重。如果路是单行道,我们就在两个顶点之间画一条有向边;如果路是双行道,我们就在两个顶点之间画两条方向不同的边。这样,整个地图就被抽象成一个有向有权图。想要解决这个问题,有一个非常经典的算法,最短路径算法,更加准确地说,是单源最短路径算法(一个顶点到一个顶点)。提到最短路径算法,最出名的莫过于 Dijkstra 算法了。

package homework.graphdelete;import java.util.Arrays;
import java.util.LinkedList;
import java.util.PriorityQueue;
import java.util.Queue;/*** 有向有权图*/
public class DirectedGraph {private int v;//顶点数量private LinkedList<Edge> adj[];public DirectedGraph(int v){this.v = v;adj = new LinkedList[v];for(int i=0;i<v;i++){adj[i] = new LinkedList<Edge>();}}public void addEdge(int sid,int tid){this.addEdge(sid,tid,1);}public void addEdge(int sid,int tid,int weight){Edge edge = new Edge(sid,tid,weight);adj[sid].add(edge);}/*** 计算从s到t的最短路径* bfs* @param s* @param t*/public void dijkstrala(int s,int t){Vertex[] vertexes = new Vertex[v];for(int i=0;i<this.v;i++){vertexes[i] = new Vertex(i,Integer.MAX_VALUE);}vertexes[s].distance = 0;boolean[] inqueue = new boolean[this.v];inqueue[s] = true;int[] predecessor = new int[this.v];Arrays.fill(predecessor,-1);PriorityQueue<Vertex> queue = new PriorityQueue<>();queue.offer(vertexes[s]);while(!queue.isEmpty()){Vertex vertex = queue.poll();if(vertex.id == t){break;}for(int j = 0;j<adj[vertex.id].size();j++){Edge edge = adj[vertex.id].get(j);Vertex to = vertexes[edge.tid];if(vertex.distance + edge.weight < to.distance){predecessor[edge.tid] = vertex.id;to.distance = vertex.distance + edge.weight;if(inqueue[to.id]){queue.remove(to);queue.offer(to);}else{queue.offer(to);inqueue[to.id] = true;}}}}printPath(predecessor,t);}private void printPath(int[] pre,int nowNode){if(pre[nowNode]!=-1){printPath(pre,pre[nowNode]);}System.out.print(nowNode+" ");}class Edge{private int sid;private int tid;private int weight;public Edge(int sid,int tid,int weight){this.sid = sid;this.tid = tid;this.weight = weight;}}class Vertex implements Comparable<Vertex>{private int id;private int distance;//从起始点到当前点的距离public Vertex(int id,int distance){this.id = id;this.distance = distance;}@Overridepublic int compareTo(Vertex o) {return this.distance - o.distance;}}
}

我们用 vertexes 数组,记录从起始顶点到每个顶点的距离(dist)。起初,我们把所有顶点的 dist 都初始化为无穷大(也就是代码中的 Integer.MAX_VALUE)。我们把起始顶点的 dist 值初始化为 0,然后将其放到优先级队列中。我们从优先级队列中取出 dist 最小的顶点 minVertex,然后考察这个顶点可达的所有顶点(代码中的 nextVertex)。如果 minVertex 的 dist 值加上 minVertex 与 nextVertex 之间边的权重 w 小于 nextVertex 当前的 dist 值,也就是说,存在另一条更短的路径,它经过 minVertex 到达 nextVertex。那我们就把 nextVertex 的 dist 更新为 minVertex 的 dist 值加上 w。然后,我们把 nextVertex 加入到优先级队列中。重复这个过程,直到找到终止顶点 t 或者队列为空。
在这里插入图片描述

时间复杂度计算:队列操作,队列中节点个数不会超过V。入队操作复杂度是O(logV)。在顶点重复插入的时候会有一个删除操作,时间复杂度O(V)。所有引起入队或者删除操作都是因为遍历到边,所有次数不会超过边的个数。总体时间复杂度最坏情况O(EV) ,最优情况O(E logV)。

3 A星算法

实际上,像出行路线规划、游戏寻路,这些真实软件开发中的问题,一般情况下,我们都不需要非得求最优解(也就是最短路径)。在权衡路线规划质量和执行效率的情况下,我们只需要寻求一个次优解就足够了。那如何快速找出一条接近于最短路线的次优路线呢?
这个快速的路径规划算法,就是我们今天要学习的 A* 算法。实际上,A* 算法是对 Dijkstra 算法的优化和改造。
Dijkstra 算法有点儿类似 BFS 算法,它每次找到跟起点最近的顶点,往外扩展。这种往外扩展的思路,其实有些盲目。为什么这么说呢?我举一个例子来给你解释一下。下面这个图对应一个真实的地图,每个顶点在地图中的位置,我们用一个二维坐标(x,y)来表示,其中,x 表示横坐标,y 表示纵坐标。
在这里插入图片描述

在 Dijkstra 算法的实现思路中,我们用一个优先级队列,来记录已经遍历到的顶点以及这个顶点与起点的路径长度。顶点与起点路径长度越小,就越先被从优先级队列中取出来扩展,从图中举的例子可以看出,尽管我们找的是从 s 到 t 的路线,但是最先被搜索到的顶点依次是 1,2,3。通过肉眼来观察,这个搜索方向跟我们期望的路线方向(s 到 t 是从西向东)是反着的,路线搜索的方向明显“跑偏”了。之所以会“跑偏”,那是因为我们是按照顶点与起点的路径长度的大小,来安排出队列顺序的。与起点越近的顶点,就会越早出队列。我们并没有考虑到这个顶点到终点的距离,所以,在地图中,尽管 1,2,3 三个顶点离起始顶点最近,但离终点却越来越远。如果我们综合更多的因素,把这个顶点到终点可能还要走多远,也考虑进去,综合来判断哪个顶点该先出队列,那是不是就可以避免“跑偏”呢?
我们设计一个函数用于估算当前节点到终点的距离。这个函数又被称为启发函数。计算两点之间的距离可以用欧几里得距离,但是欧几里得距离需要开平方,计算量大,使用曼哈顿距离代替。

A星算法与dijkstra算法不同地方在于:
1 队列中按照f值(综合计算值)排序
2 退出条件不一样,在循环中遇到终点就退出(不用查找最优路径)
3 在更新顶点dist的时候,会同步更新f值。

测试代码与实现

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

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

相关文章

spring mvc学习(10):eclipse的环境前maven配置

一.maven的安装 1解压maven压缩包到某一路径下 2配置MAVEN_HOME 3配置path到MAVEN_HOME/bin下 二eclipse集成MAVEN 方法1:直接使用自带插件 1在用户目录/.m2文件夹下面创建setting.xml文件&#xff0c;配置maven仓库位置 2在eclipse中直接配置maven的文件路径 方法2&…

for in / for of 要会用

for in是ES5标准&#xff0c;遍历index ---索引 or key --- 键. 1 for (var index in arr){} // index 0 1 2 3...arr.length-1 1 for (var key in obj){} // key obj里边enumerable的属性&#xff08;可枚举的属性&#xff09; for of是ES6标准&#xff0c;遍历value---每…

04级函授计算机等级考试练习.rar

04级函授计算机等级考试练习.rar 以下是计算机基础的练习资料:函授练习.rar posted on 2005-07-16 21:11 麦子 阅读(...) 评论(...) 编辑 收藏 转载于:https://www.cnblogs.com/maixf/archive/2005/07/16/194254.html

162. Find Peak Element

文章目录1 题目理解2 线性扫描3 递归二分查找1 题目理解 输入&#xff1a;int[] nums并且 nums[i]!nums[i1] 输出&#xff1a;找到称为峰值的那个数字&#xff0c;返回其下标。 规则&#xff1a;峰值是指&#xff1a;nums[i-1]<nums[i] 并且 nums[i1]<nums[i]。你可以认…

spring mvc学习(11):eclipse的环境maven项目创建前期工作

1创建maven项目 2pom.xml有红色错误&#xff0c;生成web目录 3修改pom.xml配置 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instan…

第二章《蜂群思维》

白痴的选举大厅&#xff0c;由白痴选白痴&#xff0c;其产生的效果却极为惊人。蜂群曾如同日蚀一样神秘、深不可测&#xff0c;他们如同被释放的精灵一样从蜂巢涌出来&#xff0c;并缓缓漂移到另一个新的世界里去。这之中没有绝对的统治者&#xff0c;他们通过侦查员的探测&…

Visual Studio.Net 快捷键表

快捷键 功能CTRL SHIFT B生成解决方案CTRL F7 生成编译CTRL O 打开文件CTRL SHIFT O打开项目CTRL SHIFT C显示类视图窗口F4 显示属性窗口SHIFT F4显示项目属性窗口CTRL SHIFT E显示资源视图F12 转到定义CTRL F12转到声明CTRL ALT J…

69. Sqrt(x)

1 题目理解 输入一个x&#xff0c;返回这个数据的平方根。如果平方跟不是整数&#xff0c;就只取整数部分。 例如x4&#xff0c;返回2&#xff1b;x8&#xff0c;返回2。 2 二分 套用模板&#xff0c;要找一个g(x)函数。g(m)m*m>x&#xff0c;找这个函数的最小值。一个数…

spring mvc学习(12)---使用idea创建第一个maven项目

一.准备条件&#xff1a; 1.安装idea旗舰版 2.安装tomcat 二.打开idea开始创建 1.创建Project 2.选择项目类型为maven 3.输入组名和项目名 ---> 下一步 ---->下一步 4.maven项目创建完成 5.运行 &#xff08;1&#xff09;配置tomcate (2)添加tomcat &#xff08;3&…

Mybaties配置一对多关系sql实例

<!-- resultMap中的type表示返回什么类型的对象 --><resultMap id"BaseGoods" type"com.cn.hjsj.pojo.BaseGoods"><!-- property 表示com.cn.hjsj.pojo.BaseGoods中的属性&#xff1b; column 表示表中的列名 --><!-- association字面…

深入浅出.NET泛型编程(1)

深入浅出.NET泛型编程(1) 使用泛型集合  .NET 2.0的System.Collections.Generics 命名空间包含了泛型集合定义。各种不同的集合/容器类都被"参数化"了。为使用它们,只需简单地指定参数化的类型即可。请看例2:例2.类型安全的泛型列表List&#xff1c;int&#xff1e…

74. Search a 2D Matrix

文章目录1 题目理解2 二分1 题目理解 输入&#xff1a;一个mxn的int数组matrix&#xff0c;这个数组每一行按照从小到大排序&#xff0c;并且下一行的第一个值大于上一行的最后一个值。一个int值target 返回&#xff1a;target在matrix中是否存在。存在返回true。 2 二分 我…

spring mvc学习(13)windows上安装maven

本地安装与配置&#xff1a; 1.jdk 在cmd中运行 Java -version 2.下载maven包 https://maven.apache.org/download.cgi下载最新版的Maven程序&#xff1a;&#xff08;选择bin zip包&#xff09; 将文件解压在D:\Program Files\apache-maven-3.5.4 3.下载eclipse并安装 …

vmware 5.51 下安装Ubuntu Linux8.04安装vmware tools不成功

得空安装Ubuntu Linux体验一把&#xff0c;记得最早还是98,99,2000年的时候玩的&#xff0c;什么冲浪(Xteam)、红帽子、红旗、蓝点&#xff0c;TurboLinux. 那时候版本都是什么1.0,2.0,3.0之类的&#xff0c; 好像Turbo Linux版本最高到4.0了。 如今已经很少有人知道蓝点Linux了…

拜拉特夜

拜拉特夜(Barat)又称“赦免之夜”&#xff0c;指伊斯兰教历舍尔巴尼月&#xff08;八月&#xff09;十五这日的夜晚。“拜拉特”意为“无辜、清白”&#xff0c;引申为“赦免”之意。这一夜安拉将更换所有人下一年命运的“文卷”&#xff0c;所以这是忏悔之夜、讨白之夜。穆斯林…

歌谣带你看java面试题

面试题来源与各个网站&#xff0c;如有不合理之处&#xff0c;欢迎吐槽。 第一题.java面试题1 牛客&#xff1a;A派生出子类B&#xff0c;B派生出子类C&#xff0c;并且在java源代码中有如下声明&#xff1a; 第二题.java面试题2 牛客:定义类中成员变量时不可能用到的修饰是 …

875. Koko Eating Bananas

文章目录1 题目理解2 二分3 相似题目10111 题目理解 输入&#xff1a;koko是个猴子&#xff0c;喜欢吃香蕉。输入int[] piles&#xff0c;表示有n堆香蕉&#xff0c;每堆香蕉有piles[i]个。koko喜欢慢慢吃香蕉&#xff0c;但还是希望在H个小时内把香蕉吃完。因为H小时之后&…

应该算是在说 delphi 的日志框架吧

最早接触 delphi 的日志是在 B 站上一个自称 老侯 的人发的视频里&#xff0c;视频名是 [delphi基础教程 第一季] 这个应该是他在B站的个人空间吧 https://space.bilibili.com/323024121 后来找 log4D 资料的时候找到了这个网站 http://www.raebear.net/ 记录 Log4D 的几句代码…

唯一

悠扬&#xff0c;流畅&#xff0c;很有韵味的感觉&#xff0c;大声对你深爱的人说你是我的唯一王力宏新专辑首支主打歌"唯一"&#xff0c;打动不少歌迷&#xff0c;觉得王力宏唱来格外有感情&#xff0c;原来是因为这首歌说的就是他自身的感动&#xff0c;凭吊一段逝…

spring mvc学习(14) the superclass “javax.servlet.http.HttpServlet” was not found

问题描述&#xff1a;   我们在用MyEclipse进行Java web开发时&#xff0c;可能会出现这样的错误&#xff1a;The superclass "javax.servlet.http.HttpServlet" was not found on the Java Build Path。我们该怎么解决这个问题呢&#xff1f; 我遇到的错误显示如下…