A Star算法总结与实现(附Demo)

关于A Star Algorithm

A star算法最早可追溯到1968年,在IEEE Transactions on Systems Science and Cybernetics中的一篇A Formal Basis for the Heuristic Determination of Minimum Cost Paths,是把启发式方法(heuristic approaches)如BFS,和常规方法如Dijsktra算法结合在一起的算法。有点不同的是,类似BFS的启发式方法经常给出一个近似解而不是保证最佳解。然而,尽管A star基于无法保证最佳解的启发式方法,A star却能保证找到一条最短路径。

公式表示为:f(n)=g(n)+h(n)

f(n)是节点n从初始点到目标点的估价函数

g(n)是在状态空间中从初始节点到n节点的实际代价

h(n)是从n到目标节点最佳路径的估计代价

观察A*寻路算法的运行轨迹

假设起点为A(浅蓝色的个字) 终点为B(深蓝色的格子)

红色代表该格子为障碍物

地图为20x20的格子

显示FGH值的格子代表经过A*算法搜索并生成路径的格子

有透明度变化的格子代表该格子有被搜索。

绿色格子代表的是搜索完成后A*得到的最优的路径

A直接抵达B的情况下

A越过直线障碍到达B

A越过U型障碍到达B

B为障碍物所包围着,A到达不了B的情况下

总结与思考

由4组图可以得到

1.A*的消耗是一个及其不稳定的过程,消耗的最小值不低于直线路径上的消耗,消耗的最大值不高于遍历整张地图的消耗。

2.A*的消耗主要在搜索的搜索格子,以及对其FGH的操作上。

3.由1,2可以得出,在对运行速率和效率有要求的场景下,A*可能不是一个比较好选择。

算法步骤

横向纵向的格子的单位消耗为10,对角单位消耗为14。

定义一个OpenList,用于存储和搜索当前最小值的格子。

定义一个CloseList,用于标记已经处理过的格子,以防止重复搜索。

开始搜索寻路

1.将起点加入OpenList

2.从OpenList中弹出F值最小的点作为当前点

3.获取当前点九空格(除去自己)内所有的非障碍且不在CloseList内的邻居点

4.遍历上一步骤得到的邻居点的集合,每个邻居点执行以下逻辑


如果邻居点在OpenList中
    计算当前值的G与该邻居点的G值
    如果G值比该邻居点的G值小
        将当前点设置为该邻居点的父节点
        更新该邻居点的GF值
若不在
    计算并设置当前点与该邻居点的G值
    计算并设置当前点与该邻居点的H值
    计算并设置该邻居点的F值
    将当前点设置为该邻居点的父节点

5.判断终点是否在OpenList中,如果已在OpenList中,则返回该点,其父节点连起来的路径就是A*搜索的路径。如果不在,则重复执行2,3,4,5。直到找到终点,或者OpenList中节点数量为0。

Tip:判定结束的有两种

第一种是以OpenList中有终点节点或者OpenList中没有节点

第二种是CLoseList中有终点节点或者......

第一种要比第二种运算次数要少许多,但在最短路径的的处理上,第二种要比第一种要精准,是相对精准。

图解算法

(7,10)为起点 ,(11,10)为终点,(9,11) (9,10)(9,9)为障碍点。

1.当前点为(7,10)

2.当前点为(8,9)

2.当前点为(8,11)

当前点为(6,10)

这里是最容易忽视的地方,因为A*的启发搜索的实现就是靠搜索F值最小的节点来实现,所以是会出现这种背离目标的搜索。

当前点为(7,9)

当前点为(7,11)

当前点为(9,8)

当前点为(10,9)

当OpenList中出现终点节点时,则结束此次搜索

如果有想看更复杂的条件下的搜索轨迹,在线的AStarDemo 或者clone github工程

总结与思考

A的消耗有很大的不确定性。消耗跟地图的复杂程度成正比,跟相对距离的长短成正比。
有一个极端的情况,当终点位置为障碍点包围时,即A Star找不到终点坐标,A
会遍历该地图此障碍区以外的所有区域。

关键逻辑的代码实现

1.A*寻路算法的主逻辑

Point start = ...;
Point end = ...;
bool isIgnoreCorner = ...;
OpenList.Add(start);
while (OpenList.Count != 0)
{
    stepSearch(start, end, isIgnoreCorner);
    if (OpenList.Get(end) != null)
        return OpenList.Get(end);
}
return OpenList.Get(end);

2.单次搜索所执行的逻辑

//找出F值最小的点
var tempPoint = OpenList.PopMinPoint();
//OpenList.RemoveAt(0);
CloseList.Add(tempPoint);
var alivePoints = GetGridAlivePoint(tempPoint, isIgnoreCorner);
for (int i = 0; i < alivePoints.Count; i++)
{
    Point p = alivePoints[i];
    if (OpenList.Exists(p))
    {
        //计算G值, 如果比原来的大, 就什么都不做, 否则设置它的父节点为当前点,并更新G和F
        FoundPoint(tempPoint, p);        
    }
    else
    {
        //如果它们不在开始列表里, 就加入, 并设置父节点,并计算GHF
        NotFoundPoint(tempPoint, end, p);
    }
}

3.当邻居点在OpenList点中时的处理逻辑

var G = CalcG(tempStart, point);
if (G < point.G)
{
    point.ParentPoint = tempStart;
    //因为每次取值,都是使用F值,所以我觉的可以不更新G值
    //point.G = G;
    point.F = point.H + G;
}

4.当邻居点不在OpenList点中时的处理逻辑

point.ParentPoint = tempStart;
point.G = CalcG(tempStart, point);
point.H = CalcH(end, point);
point.CalcF();
OpenList.Add(point);

5.最基础的逻辑也是最重要的逻辑之一,计算G值

计算G值 只适用于相邻的两个点

int G = (Math.Abs(point.X - start.X) + Math.Abs(point.Y - start.Y)) == 2 ? 14:10;
int parentG = point.ParentPoint != null ? point.ParentPoint.G : 0;
return G + parentG;

5.最基础的逻辑也是最重要的逻辑之一,计算H值

同G值,这里只计算直线上的消耗,不处理对角。

int step = Math.Abs(point.X - end.X) + Math.Abs(point.Y - end.Y);
return step * 10;

应用与思考

1.A* 在游戏中多有应用,怪物AI,计算玩家行走的路径,一些辅助工具比如游戏机器人玩家的策略方案等应用。但因为其消耗的极其不稳定,所以不会作为首选,在游戏中如果大量的应用这种逻辑,JPS(Jump Search Point),或者JPS+(JPS的优化版本)

2.A*在AR和自动驾驶领域也有应用。比如有些AR的应用是基于SLAM算法进行场景实时建模,然后在生成的模型当中,搜索一条有效的路径。A Star在这种场景中有很强的应用空间。

3.A Star的消耗主要是不断的搜索生成新的节点,不断的遍历计算。其优化思路一般也是围绕这两个点,减少搜索次数,优化遍历方案。我个人觉得JPS(Jump Point Search )就是把A Star优化做到一定程度的结果。

4.第一篇关于A Star文章是在1968年,第一篇关于JPS的文章是在2011年。在这段时间A Star处于什么样的一个地位,在这期间A Star又经历了什么样的演变,又演变出多少种在其基础之上优化的算法。在我看来刚出世时的A Star是一种算法,一种工具,在经历种种反复的推敲之后,俨然成为了一种思想,一种在未知领域寻找最优解的思想。

传送门

A*与JPS寻路算法的实现Demo

专业的各种寻路算法的Demo

我自己的WebGl Demo

AStarDemo的Github工程

参考与引用

维基百科-A* search algorithm

A星寻路算法介绍-莫水千流-博客园

A Star Algorithm总结与实现

A*多人寻路解决方案及优化策略-CSDN

posted on 2020-08-03 11:36  ThisIsRone  阅读(1305)  评论(0编辑  收藏  举报

导航