在學習A*算法以前,很好奇的是A*爲何叫作A*。在知乎上找到一個回答,大體意思是說,在A*算法以前有一種基於啓發式探索的方法來提升Dijkstra算法的速度,這個算法叫作A1。後來的改進算法被稱爲A*。*這個符號是從統計文獻中借鑑來的,用來表示相對一箇舊有標準的最優估計。git
啓發式探索是利用問題擁有的啓發信息來引導搜索,達到減小探索範圍,下降問題複雜度的目的。github
A*尋路算法就是啓發式探索的一個典型實踐,在尋路的過程當中,給每一個節點綁定了一個估計值(即啓發式),在對節點的遍歷過程當中是採起估計值優先原則,估計值更優的節點會被優先遍歷。因此估計函數的定義十分重要,顯著影響算法效率。算法
將待搜索的區域簡化成一個個小方格,最終找到的路徑就是一些小方格的組合。固然是能夠劃分紅任意形狀,甚至是精確到每個像素點,這徹底取決於你的遊戲的需求。通常狀況下劃分紅方格就能夠知足咱們的需求,同時也便於計算。
以下圖區域,被簡化成6*6的小方格。其中綠色表示起點,紅色表示終點,黑色表示路障,不能通行。
緩存
先描述A*算法的大體過程:函數
初始節點,目標節點,分別表示路徑的起點和終點,至關於上圖的綠色節點和紅色節點
F值,就是前面提到的啓發式,每一個節點都會被綁定一個F值
F值是一個估計值,用F(n) = G(n) + H(n) 表示,其中G(n)表示由起點到節點n的固定消耗,H(n)表示節點n到終點的估計消耗。H(n)的計算方式有不少種,好比曼哈頓H(n) = x + y,或者歐幾里得式H(n) = sqrt(x^2 + y^2)。本例中採用曼哈頓式。
F(n)就表示由起點通過n節點到達終點的總消耗
爲了便於描述,本文在每一個方格的左下角標註數字表示G(n),右下角數字表示H(n),左上方數字表示F(n)。具體如何計算請看下面的一個例子學習
接下來,咱們嚴格按照A*算法找出從綠色節點到紅色節點的最佳路徑
首先將綠色節點加入到open列表中
接着判斷open列表不爲空(有起始節點),紅色節點不在open列表中
而後從open列表中取出F值最小的節點,此時,open列表中只有綠色節點,因此將綠色節點取出,做爲當前節點,並將其加入到close列表中
計算綠色節點的相鄰節點(暫不考慮斜方向移動),以下圖所示的全部灰色節點,並計算它們的F值。這些子節點既沒有在open列表中,也沒有在close列表中,因此都加入到open列表中,並設置它們的父節點爲綠色節點優化
F值計算方式:
以綠色節點右邊的灰色節點爲例
G(n) = 1,從綠色節點移動到該節點,都只須要消耗1步
H(n) = 3,其移動到紅色節點須要消耗橫向2步,豎向一步,因此共消耗3步(曼哈頓式)
F(n) = 4 = G(n) + H(n)lua
試着算一下其餘灰色節點的F值吧,看看與圖上標註的是否一致3d
繼續選擇open列表中F值最小的節點,此時最小節點有兩個,都爲4。這種狀況下選取哪個都是同樣的,不會影響搜索算法的效率。由於啓發式相同。這個例子中按照右下左上的順序選取(這樣能夠少畫幾張圖(T▽T))。先選擇綠色節點右邊的節點爲當前節點,並將其加入close列表。其相鄰4個節點中,有1個是黑色節點不可達,綠色節點已經被加入close列表,還剩下上下兩個相鄰節點,分別計算其F值,並設置他們的父節點爲黃色節點。blog
此時open列表中F值最小爲4,繼續選取下方節點,計算其相鄰節點。其右側是黑色節點,上方1號節點在close列表。下方節點是新擴展的。主要來看下左側節點,它已經在open列表中了。根據算法咱們要從新計算它的F值,按通過2號節點計算H(n) = 3,G(n)不變,因此F(n) = 6相比於原值反而變大了,因此什麼也不作。(後面的步驟中從新計算F值都不會更小,再也不贅述)
此時open列表中F值最小仍爲4,繼續選取
此時open列表中F值最小爲6,優先選取下方節點
此時open列表中F值最小爲6,優先選取右方節點
此時open列表中F值最小爲6,優先選取右方節點
此時open列表中F值最小爲6,優先選取右方節點
此時咱們發現紅色節點已經被添加到open列表中,算法結束。從紅色節點開始逆推,其父節點爲7號,7號父節點爲6號,6號父節點爲5號…,最終獲得檢索路徑爲:綠色-1-2-5-6-7-紅色
在上面的例子中,全部遇到已經在open列表中的節點從新計算F值都不會更小,沒法作更新操做。
因此再舉一個例子來演示這種狀況。相同的搜索區域,假設豎向或橫向移動須要消耗1,此次也支持斜方向移動了,可是斜方向可能都是些山路很差走,移動一次須要消耗4。對應的相鄰節點F值以下圖所示
一樣選擇open列表中F值最小的節點,咱們優先選擇了右方節點,計算其相鄰節點。共8個。其中三個是黑色節點,一個綠色節點在close列表中,不考慮。上方兩個和下方兩個都是已經在open列表中了,要從新計算F值。
先看左上角的相鄰節點,經過黃色節點到達改節點,H(n) = 5,G(n)不變,F(n)反而更大了,因此什麼也不作。左下角節點同理。
上方居中節點,經過黃色節點計算H(n) = 2, G(n)不變,F(n) = 6 < 8 因此,更新這個節點的F值,並將其父節點修改成黃色節點。下方居中節點同理。
寫了一套A*算法的Lua實現。主要特色以下:
優化效率,採用了map緩存,避免屢次循環遍歷
支持配置移動權重
支持配置是否能夠斜向移動,斜向時牆角是否可通行
源碼請查看:https://github.com/iwiniwin/LuaKit/blob/master/AStar.lua