-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathastar.ss
64 lines (62 loc) · 2.91 KB
/
astar.ss
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
(export #t)
(import
(only-in :std/assert assert!)
(only-in :std/iter for)
(only-in :std/misc/hash hash-ensure-ref)
(only-in :std/misc/pqueue make-pqueue pqueue-empty? pqueue-peek pqueue-pop! pqueue-push!)
(only-in :std/srfi/1 append-reverse)
(only-in :std/sugar hash until))
;; A* from start, exhaustively collect all the best paths to all other points.
;; Assume cost is real-valued.
(def (A* starts: starts arcs: arcs bests: (bests (let (h (hash)) (cut hash-ensure-ref h <> <>))))
(def q (make-pqueue car)) ;; queue of forward nodes
(for-each (cut pqueue-push! q <>) starts)
(until (pqueue-empty? q)
(let (csp (pqueue-pop! q))
(with ([cost state . path] csp)
(bests state (lambda ()
(for (([co . st] (arcs state)))
(pqueue-push! q [(+ cost co) st state . path]))
csp)))))
bests)
;; A* from both start and end, find out the best path
(def (A*2 starts: starts ends: ends +arcs: +arcs -arcs: -arcs) ;; assume cost is real-valued
(def +q (make-pqueue car)) ;; queue of forward nodes
(def -q (make-pqueue car)) ;; queue of backward nodes
(for-each (cut pqueue-push! +q <>) starts)
(for-each (cut pqueue-push! -q <>) ends)
(def +t (hash)) ;; table of forward states with best cost
(def -t (hash)) ;; table of backward nodes with best cost
(def best [+inf.0 . #f]) ;; cost to beat (once complete paths have been made)
(def mostcost 0) ;; most cost for an arc
;; We could be more precise by instead remembering the least cost
;; to a state *preceding* an active path, but that would require more bookkeeping.
(def (search queue table -table arcs forward?)
(with ([cost state . path] (pqueue-pop! queue))
(cond
((hash-get table state) ;; previous path to that state
=> (lambda (prev) (assert! (<= (car prev) cost)))) ;; best-path first, so previous must be optimal
((hash-get -table state) ;; found a connection!
=> (match <> ([c . p]
(let (total (+ cost c))
(when (< total (car best))
;;(DBG FOUND!: 'c cost '-c c 'p path '-p p forward?)
(let (path (append-reverse path p)) ;; NB: state already in p
(set! best [total :: (if forward? path (reverse path))])))))))
(else
(hash-put! table state [cost state . path])
(for (([co . st] (arcs state)))
(when (> co mostcost) (set! mostcost co))
(pqueue-push! queue [(+ cost co) st state . path]))))))
(let loop ()
(if (or (pqueue-empty? +q) (pqueue-empty? -q))
best
(let ((+best (car (pqueue-peek +q)))
(-best (car (pqueue-peek -q))))
(if (<= (car best) (- (+ +best -best) mostcost))
best
(begin
(if (<= +best -best)
(search +q +t -t +arcs #t)
(search -q -t +t -arcs #f))
(loop)))))))