A*算法改进——Any-Angle Path Planning的Theta*算法与Lazy Theta*算法

A*算法改进——Any-Angle Path Planning的Theta*算法与Lazy Theta*算法

大家好,又见面了,我是全栈君。

本文是该篇文章的归纳http://aigamedev.com/open/tutorial/lazy-theta-star/#Nash:07 。

871155-20170208112240760-1421237203.png

传统的A*算法中,寻找出来的路径只能是沿着给出的模型(比如TileMap、网格模型)上的路径依次行走。(上图上半)

在游戏中寻路的场合下,这种路径看起来是十分不自然的。与之相对的“自然的”寻路称为Any-Angle Path Planning。(上图下半)

对于A*的一个简单的修正是,在生成路径之后,检测路径上隔开的两个点之间是否有line of sight(即视线,以下简称LOS),有的话便可以把中间的点删除,让人物直接穿过。这种方法称之为A* with post-smoothed paths。依据如何选取这两个点,最终优化的效果不同。但是要注意的是这种修正得到的路径的质量和效率是需要取舍的。

另外,本文介绍的算法都不保证寻找的路径是最佳的。不要尝试用来刷题233。

Theta*

修正的第二种办法,即Theta*算法。这种算法是A*的一种改进,关键在于其打开一个节点s,然后更新周围的节点s’时,会检查s’与parent(s)的可见性。如果可见,则把s’的父节点设置成parent(s)。

871155-20170208111318807-1517171879.png
左边是\A*算法,中间是Theta*算法,右边是等下介绍的Lazy Theta*。

可以看到在Theta*中,除了ComputeCost函数之外,其他的内容和A*算法是相同的。

在ComputeCost中,A*算法计算一个点s’的新的g,如果比原来的更好,则将原来的parent和g换成新的。

在Theta*中,则在计算前,先去计算s’s的parent有无视线。如果有视线,则把s’的parent和gs的parent为parent进行更新。否则跟A*一样处理。

然而,Theta*有一个很大的问题,就是需要做大量的line of sight检查。有多少个点进入过open列表,就有多少次检查。在较为细致的网格中这个数量是十分巨大的。

Lazy Theta*

这里就引出了Theta*的一种优化,即Lazy Theta*。

871155-20170208112051369-758210801.png
两种算法进行的LOS检查数。

Lazy Theta*的核心思想在于,将line of sight检查延迟到打开该节点为止。

871155-20170208112425354-1523539550.png
示例,注意右上图(第二步)中B2指向的是start而不是B3,因为B2尚未打开,我们可以乐观认为B2和父节点B3的父节点有LOS。这一假设在左下图(第三步)中,B2打开时才得到修正。

在Theta*中,检查视线的时机发生于一个点进入open列表的时候。但是实际上,有很多进入open列表的点最终不在路径中,这意味着LOS检查是无效的。因此Lazy Theta*选择把LOS检查放到打开该节点的时候进行。

当然,进入open列表的点,我们需要设置它的g值和parent。在Lazy Theta*中,我们乐观地认为在这里LOS检查永远成立的,因此g值和parent值得的设置按照Theta*中LOS检查成立一般地进行设置。

随后,在打开一个节点时,我们对这个点调用SetVertex方法。该方法中我们对该点和它的父节点进行真正的LOS检查。如果成立,那么我们之前的假设是对的,那就继续进行下去。如果没有LOS,那么我们还需要为这个点找到一个正确的parent。

对点s找到正确的parent的方法稍微繁杂一点。首先我们要知道什么点能成为s的parent,答案是在close表里的点。其次还应该是和s有LOS关系的点。如果有的选择的话,我们还要选择g(parent) + c(parent,s)最小的。

当然,如果我们对close表里的点全部做一遍LOS检查那就是本末倒置了。在这里我们直接取s的相邻点作为parent的候选。然后和close表做个交集,在其中选择最好的点即可。

Lazy Theta*的优化

这个优化是在A*中就存在的,优化的A*叫做Weighted A*,即带权重的A*。

所谓权重是加给启发函数的。原本的估值函数F(s) = G(s) + H(s),加入权重后变成F(s) = G(s) + weight * H(s) 。

这意味着比起从出发点到当前点的距离,当前点到终点的距离的估值影响更大(当weight > 1时)。

可以简单的理解为weight越大,会“更想接近终点”。

在Lazy Theta*中也可以直接套用这个优化。同时比起普通的weighted A*,Lazy Theta*中的这项优化更加有价值。具体的就不在这里叙述了,在原文中有提到。

下一篇文章中我会简单叙述一下我在unity/C#中实现的Lazy Theta*寻路。

转载于:https://www.cnblogs.com/yangrouchuan/p/6373285.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/108671.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • Python虚拟环境(pipenv、venv、conda一网打尽)[通俗易懂]

    Python虚拟环境(pipenv、venv、conda一网打尽)[通俗易懂]随着大数据、人工智能的兴起,Python被带到了一个新的高度,但在使用Python过程中,很多人没搞清楚Python环境究竟是什么。当开发工程的时候,往往因为python环境的问题搞得一团糟。本文旨在说清楚什么是Python环境,什么是Python虚拟环境,并希望通过本文的学习掌握常用的Python环境管理工具的使用。

    2022年8月27日
    9
  • 史上最全的正则表达式-匹配中英文、字母和数字

    史上最全的正则表达式-匹配中英文、字母和数字在做项目的过程中,使用正则表达式来匹配一段文本中的特定种类字符,是比较常用的一种方式,下面是对常用的正则匹配做了一个归纳整理。1、匹配中文:[\u4e00-\u9fa5]2、英文字母:[a-zA-Z]3、数字:[0-9]4、匹配中文,英文字母和数字及下划线:^[\u4e00-\u9fa5_a-zA-Z0-9]+$同时判断输入长度:[\u4e00-\u9fa5_a-zA-Z0-9_…

    2022年9月25日
    4
  • 智能小车设计规划_智能循迹避障小车设计

    智能小车设计规划_智能循迹避障小车设计摘要该课题主要基于单片机的循迹、避障、WiFi、蓝牙等功能的智能小车,在一些特殊环境下有着特殊的意义。硬件控制以arduino为控制核心。采用超声波避障和红外避障传感器共同完成寻迹、避障功能,并将相关信号传送给单片机,经单片机控制系统分析判断后控制驱动芯片驱动直流电机实现小车前进、后退、左转、右转,停止。软件采用移植性较好的c语言编写,通过手机蓝牙App实现对智能小车的控制。通过TCP/UD协…

    2022年10月18日
    2
  • 详解Java中的clone方法

    详解Java中的clone方法

    2022年3月1日
    46
  • websocket大文件发送(分片传送思想)

    websocket大文件发送(分片传送思想)目前的项目是在做一款带桌面共享的代码编辑器,其中需要一个发送大文件的功能,传统的node.js处理大文件就是用Buffer.slice(0.offset)的思路把文件分割开,然后通过tcp或udp分开发送。前端中处理二进制的有Blob,它也有slice的方法,也可以将文件拆分开。然后借助websocket发开发送,然后在客户端(注意不是服务端)将文件合并。有人说websocket可以直接发,但是他的大小受到限制,比如发200M的东西,就会出问题。而我的方案就不会存在问题.最主要的是在发送文件的同时也不会影响

    2022年7月11日
    81
  • 23种常用设计模式的UML类图

    23种常用设计模式的UML类图23种常用设计模式的UML类图本文UML类图参考《HeadFirst设计模式》(源码)与《设计模式:可复用面向对象软件的基础》(源码)两书中介绍的设计模式与UML图。整理常用设计模式的类图,一

    2022年6月30日
    29

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号