Skip to content

Instantly share code, notes, and snippets.

@zmaril
Created August 8, 2011 02:15
Show Gist options
  • Save zmaril/1131095 to your computer and use it in GitHub Desktop.
Save zmaril/1131095 to your computer and use it in GitHub Desktop.
astar search
(def world [[ 1 1 1 1 1]
[999 999 999 999 1]
[ 1 1 1 1 1]
[ 1 999 999 999 999]
[ 1 1 1 1 1]])
(defn estimate-cost [step-cost-est size y x]
(* step-cost-est (- (+ size size) y x 2)))
(defn path-cost [node-cost cheapest-nbr]
(+ node-cost
(:cost cheapest-nbr 0)))
(defn total-cost [newcost step-cost-est size y x]
(+ newcost
(estimate-cost step-cost-est size y x)))
(defn min-by [f coll]
(when (seq coll)
(reduce (fn [min this]
(if (> (f min) (f this)) this min)) coll)))
(defn astar [start-yx step-est cell-costs]
(let [size (count cell-costs)]
(loop [steps 0
routes (vec (replicate size (vec (replicate size nil))))
work-todo (sorted-set [0 start-yx])]
(if (empty? work-todo)
[(peek (peek routes)) :steps steps]
(let [[_ yx :as work-item] (first work-todo)
rest-work-todo (disj work-todo work-item)
nbr-yxs (neighbors size yx)
cheapest-nbr (min-by :cost
(keep #(get-in routes %)
nbr-yxs))
newcost (path-cost (get-in cell-costs yx)
cheapest-nbr)
oldcost (:cost (get-in routes yx))]
(if (and oldcost (do (println (str newcost oldcost)) (>= newcost oldcost)))
(recur (inc steps) routes rest-work-todo)
(recur (inc steps)
(assoc-in routes yx
{:cost newcost
:yxs (conj (:yxs cheapest-nbr [])
yx)})
(into rest-work-todo
(map
(fn [w]
(let [[y x] w]
[(total-cost newcost step-est size y x) w]))
nbr-yxs)))))))))
(astar [0 0] 900 world)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment