路径规划——搜索算法详解(七):D*lite算法详解与Matlab代码

!!!注意!!!

看本篇之前,一定要先看笔者上一篇的LPA*讲解,笔者统一了符号看起来过渡会更加好理解!

到目前为止,我们学习了广度优先搜索Dijkstra算法、能够计算任意两个路径点距离的Floyd算法、基于采样实现快速计算的RRT算法、能够利用启发项加速探索过程的A*算法、适用于动态环境反向搜索的D*算法、增量式动态搜索的A*——LPA*算法,每种算法都有其特点,大家根据需求选择就好。

无独有偶,上述6种路径搜索算法仅适用于处理起点与终点固定情况下的路径搜索问题,本文将介绍搜索算法篇的最后一种算法——D*lite算法,该算法基于D*反向搜索的思想、LPA*增量式搜索的思想,是一种变起点反向增量式动态路径搜索算法,接下来便进入正题吧,老规矩分为算法介绍与案例讲解。(PS:搜索算法部分总结完后将开启路径优化的篇章,将介绍各种拟合轨迹的曲线,并且给出能够直接采用的C++代码,需要的朋友可以关注后续!)

一、D*lite算法流程:

1.与LPA*算法的联系与区别:

其算法原理与LPA*算法类似(没有看的朋友先看我上一篇LPA*算法的讲解再来看,因为里面定义的符号含义需要理解,不然看不懂的!!!),其改进了LAP*中的代价项k,增加了机器人当前实际移动距离km(机器人位置移动可以看作是起点位置移动了),其定义如下:

如上图所示S(黄色)为当前遍历节点,G(红色)为终点位置,绿色为机器人所在的位置(可以看作是路径搜索的起点),如左图所示,g(s)为终点距离所记录的最小代价值,rhs(s)为搜索过程中所更新计算的当时的终点距离代价值,对角移动距离为1.4,直接连接距离为1,通过计算即可得到对应的值。当机器人移动到新的s_star时原位置点标记为s_last,此时h变为1,km更新为1,左右图k1值保持不变,计算结果如上图2所示。

上述就是k值更新过程中与LPA*算法的区别,此外流程上相差并不大,大致框架相同,稍后会进行讲解。但由于k值计算方式的不同会产生新的问题:机器人每移动一步,km实时发生变化,此时对于每一个节点都需要更新其k1值。当当前优先队列U中节点较多的时候,每运动一步都需要更新,这会增加计算负担。

本算法提出了一种解决方案,当机器人起点位置发生改变后,与其减去原优先队列U中每一个节点的h值,不如直接将新插入队列的节点加上km值(如上两个图所示,h减少的数值正好等于km所增加的数值),这样就不会影响U节点中的弹出顺序,并且节省计算资源。

2.算法代码框图:

符号含义如下所示:

  • g(s):之间记录的起点距离代价的最小值;
  • rhs(s):基于父节点g所预测的最小值,设s'为s的父节点,此时有rhs(s)=g(s')+d(ss'),d(ss')表示s与父节点s‘的连接代价,由于D*lite为反向搜索的算法,所以父节点指的是离终点近的路径点。
  • K:代价值,与LPA*不同,其K(包含k1、k2)计算如下所示:

  • S:地形图中路径点集合
  • U:优先队列,每次弹出k值最小的节点s;

其函数包含初始化部分(Initialize)、计算路径部分(ComputeShortestPath)、更新节点部分(UpdateVertex)、计算代价部分(CalculateKey),直接看估计会蒙圈,下面将通过一个例子讲解该算法的更新过程

3.D*lite算法案例讲解:

1.地图介绍:

搜索地图如上图所示,其中B1为起点,E3为目标点,在地图预处理时,根据设定的启发函数(如曼哈顿距离、欧氏距离、切比雪夫距离)等计算出每一个点的启发值h,规定直线距离、对角移动距离均为1。

2.初始化(起点可以看作是第1次循环):

初始化代码如Initialize函数(02-06行)所示,首先将优先队列U初始化为空,初始化机器人运动距离km=0(即机器人当前没有发生移动,起点没有发生运动)。初始化所有节点集合S中的所有节点终点最小距离g(s)与实时计算终点距离rhs(s)为无穷大。将起点rhs置为0,并且将其插入到优先队列中U中启动搜索。

此时U={E3},此时g(E3)=∞、rhs(E3)=0、h(E3)由上一步计算为3,此时k(E3)=[0+3+0,0]=[3,0],接下来执行代码中的步骤2——ComputeShortestPath(代码10-20行),进图while循环(当且仅当搜索起点达到局部一致弹出的k值大等于起点的k值时才跳出循环

首先从U中弹出k值最小的节点,此时U中只有E3,此时CalculateKey(E3)=[3,0]=kold(E3),跳到代码15行进行判断,得到g(E3)=∞ > rhs(E3)=0,满足条件,执行16-17行的代码,此时更新g(E3)=rhs(E3)=0,E3达到局部平衡,标记为0。此外,遍历所有周围节点中的父节点(此处规定距离起点近的为父节点),即D2、E2、D3。

3.拓展节点(第2次循环):

遍历D2、E2、D3分别带入UpdateVertex函数中(代码07-09行),根据其子节点E3更新rhs值:如rhs(D2)=1(D2与E3的连接代价)+g(E3)=1+0=1,同样可以计算的到rhs(E2)=1、rhs(D3)=1,判断其是否为局部一致(08行代码),由于初始化g值均为∞,D2、E2、D3均为局部过一致状态,此时计算其k值,并将其加入到优先级队列U中,这步结束后U={D2[3,1],E2[4,1],D3[3,1]}。

4.拓展节点(不断循环,得到起点没有变化时的最终路径):

不断循环(上面的步骤跟LPA*算法几乎一样,只是搜索方向上是反过来的,LPA*从起点开始而D*lite从终点开始,还是不难理解的) ,得到如上图所示的结果,此时满足循环跳出的条件,得到最终路径为B1-C1-D2-E3。跳出步骤2,主函数中的ComputeShortestPath部分执行结束(代码23行)

5.起点变换时候的处理:

如左图所示,此时机器人位置由B1变为了C1,D2变为了障碍物,执行24-35行的代码。

当起点与终点不为同一节点时候进入该循环,首先更新所有节点与当前新起点的启发值,计算新的km值km(new)=km+h(s_start,s_last)=0+1=1(代码第30行),此时检查受影响的节点,扩散其变化造成的影响(代码32-34行),详细执行更新代价步骤如下所示:

如图6-1所示,D2变为障碍物rhs(D2)=inf,由于C1与D2直接相连(代码第32行),C1的rhs值受到影响变rhs(C1)=d(D1与C1之间的距离)+g(D1)=1+inf=inf。同理找出C1的子节点,并更新其rhs值,rhs(D1)=rhs(E1)=inf,这便是更新障碍物的传递过程(代码第33行)。

此时C1检查周围节点,查找能够将rhs(C1)值变为最小的节点作为父节点(代码第34行,跳到步骤4),此时将B1作为父节点能够使得rhs(C1)=3+1=4,与g(C1)=2不相等为局部欠一致,将其重新插入到队列U中(代码09行) ,同理将D1、E1也加入到队列U中。

更新完所有受到影响的节点后,再次调用路径搜索函数(代码第35行)重复2-4的步骤,即可得到新的路径为C1-D1-E2-E3。

该算法的奇妙之处在于在起点更新后更新了h值,能够启发新一轮的搜索,大家可以这样理解,一旦节点的起点发生变化,基于起点求解得到的每一个节点的h值肯定会发生改变,这工作量是巨大的,这是LPA*算法不能用于变起点搜索的主要原因。为了减少计算量,我们无需在原有的节点上减去h值,而是维护一个由于起点位置更改而受影响km值,km值可以理解为我们原本需要更新在原来每一个节点上h值的差值,原本我们需要在每一个节点h中加或者减km,此时我们只需要在新更新的节点上减或者加上这个差额,维持其相对的大小顺序,妙哉!

二、D*lite算法MATLAB代码

这是笔者按照古月的课程打出来的代码,已经上传到了本人Github,需要的自取:

Adamaser/Path-Planning (github.com)

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

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

相关文章

基于深度学习的日常场景下的人脸检测系统(网页版+YOLOv8/v7/v6/v5代码+训练数据集)

摘要:本文详细介绍基于YOLOv8/v7/v6/v5的日常场景下的人脸检测,核心采用YOLOv8并整合了YOLOv7、YOLOv6、YOLOv5算法,进行性能指标对比;详述了国内外研究现状、数据集处理、算法原理、模型构建与训练代码,及基于Streaml…

AWS-EKS 给其他IAM赋予集群管理权限

AWS EKS 设计了权限管理系统,A用户创建的集群 B用户是看不到并且不能管理和使用kubectl的,所以我们需要共同管理集群时就需要操场共享集群访问给其他IAM用户。 两种方式添加集群控制权限(前提:使用有管理权限的用户操作&#xff…

【c++】类和对象(七)

&#x1f525;个人主页&#xff1a;Quitecoder &#x1f525;专栏&#xff1a;c笔记仓 朋友们大家好&#xff0c;本篇文章来到类和对象的最后一部分 目录 1.static成员1.1特性 2.友元2.1引入&#xff1a;<<和>>的重载2.2友元函数2.3友元类 3.内部类4.匿名对象5.拷…

聚观早报 | 蔚来推出油车置换补贴;iPhone 16 Pro细节曝光

聚观早报每日整理最值得关注的行业重点事件&#xff0c;帮助大家及时了解最新行业动态&#xff0c;每日读报&#xff0c;就读聚观365资讯简报。 整理丨Cutie 4月02日消息 蔚来推出油车置换补贴 iPhone 16 Pro细节曝光 小米SU7创始版第二轮追加开售 OpenAI将在日本设立办事…

Ollama教程——入门:开启本地大型语言模型开发之旅

Ollama教程——入门&#xff1a;开启本地大型语言模型开发之旅 引言安装ollamamacOSWindows预览版LinuxDocker ollama的库和工具ollama-pythonollama-js 快速开始运行模型访问模型库 自定义模型从GGUF导入模型自定义提示 CLI参考创建模型拉取模型删除模型复制模型多行输入多模态…

ADB 命令之 模拟按键/输入

ADB 命令之 模拟按键/输入 模拟按键/输入 在 ​​adb shell​​​ 里有个很实用的命令叫 ​​input​​&#xff0c;通过它可以做一些有趣的事情。 ​​input​​ 命令的完整 help 信息如下&#xff1a; Usage: input [<source>] <command> [<arg>...] Th…

SV学习笔记(一)

SV&#xff1a;SystemVerilog 开启SV之路 数据类型 內建数据类型 四状态与双状态 &#xff1a; 四状态指0、1、X、Z&#xff0c;包括logic、integer、 reg、 wire。双状态指0、1&#xff0c;包括bit、byte、 shortint、int、longint。 有符号与无符号 &#xff1a; 有符号&am…

云计算对象存储服务

对象存储服务&#xff08;OSS&#xff09;中的存储桶(Bucket)叫做‘OBS桶 存储桶&#xff08;Bucket&#xff09;&#xff1a;存储桶式对象存储服务中用于存储对象的基本容器&#xff0c;类似于文件系统中的文件夹。每个存储桶具有唯一的名称&#xff0c;并且可以在桶中存储任…

理解Three.js的相机

大家都知道我们生活中的相机&#xff0c;可以留下美好瞬间。那Three.js的相机是什么呢&#xff1f;Three.js创建的场景是三维的&#xff0c;而我们使用的显示器显然是二维的&#xff0c;相机就是抽象的定义了三维空间到二维显示器的投影方式。Three.js常见的相机有两类&#xf…

【Java+Springboot】----- 通过Idea快速创建SpringBoot项目操作方法

一、第一步&#xff1a; 点击选择【File】->【New】-> 【Project】 最后弹出[new Project]界面。 二、第二步&#xff1a; 1. 选择【Spring Initializr】 2. 然后选择【Project SDK】的版本 3. 然后 Choose Initializr Service URL 选择默认&#xff08;Default&#x…

华为审核被拒提示: 您的应用存在(最近任务列表隐藏风险活动)的行为,不符合华为应用市场审核标准

应用审核意见&#xff1a; 您的应用存在&#xff08;最近任务列表隐藏风险活动&#xff09;的行为&#xff0c;不符合华为应用市场审核标准。 修改建议&#xff1a;请参考测试结果进行修改。 请参考《审核指南》第2.19相关审核要求&#xff1a;https://developer.huawei.com/c…

uniapp-设置UrlSchemes从外部浏览器H5打开app

需求&#xff1a;外部浏览器H5页面&#xff0c;跳转到uniapp开发的原生app内部。 1、uniapp内部的配置&#xff1a; &#xff08;1&#xff09;打开manifest->App常用其他设置&#xff0c;如下&#xff0c;按照提示输入您要设置的urlSchemes&#xff1a; &#xff08;2&am…

redis 的StringRedisTemplate

6.3 StringRedisTemplate 尽管JSON的序列化方式可以满足我们的需求&#xff0c;但依然存在一些问题&#xff0c;如图&#xff1a; 为了在反序列化时知道对象的类型&#xff0c;JSON序列化器会将类的class类型写入json结果中&#xff0c;存入Redis&#xff0c;会带来额外的内存…

create table main (num int);insert into main values

create table main (num int); insert into main values (3), (12), (15), (25), (23), (29), (34), (37), (32); int num1 3; int num2 12; int num3 23; 把num值处于[20&#xff0c;29]之间&#xff0c;改为20 num值处于[30,39]之间的&#xff0c;改为30 思…

C++算法补充---STL

这里写目录标题 CSTL容器字符串函数(string容器函数)字符串转字符 算法交换函数拿到容器或者数组的第一个最大&#xff08;小&#xff09;值元素的下标或者值排序函数求字符数组的有效长度atoi函数&#xff08;将字符串类型的数字转为真正的int型数字&#xff09;string转字符 …

ES学习日记(八)-------ik安装和简易使用

一、下载和安装 https://github.com/infinilabs/analysis-ik.git 网络不好可以用这个地址,注意:ik版本要和es版本保持一致 现成地址 注意es用户操作或给es用户权限 plugins新建ik文件夹,并把压缩包解压到ik unzip elasticsearch-analysis-ik-7.4.2.zip /bin目录启动es: 二…

GPT4不限制使用次数了!GPT5即将推出了!

今天登录到ChatGPT Plus账户&#xff0c;出现了如下提示&#xff1a; 已经没有了数量和时间限制的提示。 更改前&#xff1a;每 3 小时限制 40 次&#xff08;团队计划为 100 次&#xff09;&#xff1b;更改后&#xff1a;可能会应用使用限制。 GPT-4放开限制 身边订阅了Ch…

elment UI el-date-picker 月份组件选定后提交后台页面显示正常,提交后台字段变成时区格式

需求&#xff1a;要实现一个日期的月份选择<el-date-picker :typeformData.dateType :value-formatdateFormat v-modelformData.leaveFactoryDateplaceholder选择月份></el-date-picker>错误示例&#xff1a;将日期显示类型(type)dateType或将日期绑定值的格式(val…

LabVIEW专栏三、探针和断点

探针和断点是LabVIEW调试的常用手段&#xff0c;该节以上一节的"测试耗时"为例 探针可以打在有线条的任何地方&#xff0c;打上后&#xff0c;经过这条线的所有最后一次的数值都会显示在探针窗口。断点可以打在程序框图的所有G代码对象&#xff0c;包括结构&#xf…

【stm32】USART编码部分--串口数据包

USART串口数据包【源码放在最后】 关于数据包的分类 关于数据包的发送 如果想要发送Hex数据包&#xff0c;定义一个数组填充数据&#xff0c;然后使用串口模块函数SendArray进行发送 如果想要发送文本数据包&#xff0c;写一个字符串然后调用SendString进行发送 对于发送数据…