A* поиск пути на 2D-сетке не находит оптимальный путь

Я пытаюсь реализовать алгоритм A* на двумерной квадратной сетке. Тем не менее, он почти никогда не находит оптимальный путь, и я просто не понимаю, почему. Код также подозрительно медленный, даже для Python.

Я пробовал что угодно, у меня нет идей.

Это то, что я до сих пор:

файл astar.py:

end = None

NLUT = [ (1,0) , (0,1) , (-1,0) , (0,-1) ]

class Tile:
    def __init__(self,x,y,g=0,parent=None):
        self.x = x
        self.y = y
        self.g = g
        self.parent = parent

    def __eq__(self,other):
        if other == None:
            return False
        return ( (self.x == other.x) and (self.y == other.y))
    def __ne__(self,other):
        return not self.__eq__(other)
    def __hash__(self):
        return hash((self.x,self.y))

    def __str__(self):
        if self.parent == None:
            sss = ""
        else:
            sss = " <- "+str(self.parent.coords())
        return "<"+str(self.x)+"."+str(self.y)+"g"+str(self.g)+">"+sss
    def __repr__(self):
        return "<"+str(self.x)+"."+str(self.y)+">"

    def coords(self):
        return (self.x,self.y)

    def f(self):
        return self.g + self.h()

    def h(self):
        global end
        return ((abs(self.x-end[0]) + abs(self.y-end[1])))

    def nbd(self):
        global NLUT
        return [ Tile(self.x + n[0], self.y + n[1], self.g + 1, self) for n in NLUT]



def pathfind(mapfunction,start,endp):

    global end
    end = endp
    DEBUG = False
#   DEBUG = True


    def log(st):
        if DEBUG:
            print(st)

    startile = Tile(start[0],start[1],0)
    endtile = Tile(end[0],end[1])

    path = []

    # init openSet with the starting tile
    openSet = set([startile])
    closedSet = set()

    stepcount = 0

    #the loop, as long as openSet is not empty:
    while len(openSet)>0:

        log(str(stepcount)+"-"+str(openSet))
        stepcount += 1
        fcur = float("inf")
        #find lowest f-count tile in the open set
        for o in openSet:
            if o.f() < fcur:
                fcur = o.f()
                current = o

        log("current: "+str(current))

        #move the current tile to the closed set
        openSet.remove(o)
        closedSet.add(o)

        #find the von neumann neighbourhood
        nbd = o.nbd()
        log("nbd: "+str(nbd))
        #work on the neighbours
        for n in nbd:

            log("processing "+str(n))


            if mapfunction(n.x,n.y) != 0: #if it's blocked, ignore
                log("it was blocked.")
                continue
            elif n in closedSet: #if it's in the closed set, ignore
                log("it was in the closed set.")
                continue
            elif n in openSet: #if it's in the open set...
                log("it's in the open set...")
                for e in openSet: 
                    if n==e:            #find the old copy of n in the open set
                        if n.g < e.g:   #if it's a better path, substitute
                            log("SUBSTITUTION")
                            openSet.remove(e)
                            openSet.add(n)
                        else:
                            log("old path was better.")

                        break           #no need to go on...
            else: #if it's not in the open set, add it
                log("not in the open set, adding...")
                openSet.add(n)


        if endtile in closedSet:          #if we're done
            #find copy of endtile in closedSet
            for e in closedSet:
                if e == endtile:
                    rec = e
            while (not rec == None) and rec != startile:  #reconstruct the path
                path.append(rec.coords())
                rec = rec.parent
            path.append(startile.coords())
            return path  
    #if openset is empty, no path was found. :(        
    return -1

и это небольшая демонстрационная программа astardemo.py:

import astar
from random import *

mapp = [[randint(0,10)//10 for _ in range(0,50)] for _ in range(0,50)]

def isblocked(x,y):
    global mapp

    if not (x in range(0,50) and y in range(0,50)):
        return 1
    return (mapp[x][y] != 0)

start = (randint(0,49),randint(0,49))
end = (randint(0,49),randint(0,49))

mapp[start[0]][start[1]] = 0
mapp[end[0]][end[1]] = 0

p = astar.pathfind(isblocked,start,end)

print p

l = ""

for i in range(0,50):
    for j in range(0,50):
        if (i,j) in p:
            if mapp[i][j] > 0:
                l+="E"
            else:
                l+=str(p.index((i,j))%10)

        elif mapp[i][j] > 0:
            l+="#"
        else:
            l+=" "
    l+="\n"
print l

1 ответ

    for o in openSet:
        if o.f() < fcur:
            fcur = o.f()
            current = o

    log("current: "+str(current))

    #move the current tile to the closed set
    openSet.remove(o)
    closedSet.add(o)

    #find the von neumann neighbourhood
    nbd = o.nbd()

В этом коде вы используете o но вы находитесь вне цикла, поэтому оно всегда будет последним значением в цикле, которое вы намеревались использовать current?

    for o in openSet:
        if o.f() < fcur:
            fcur = o.f()
            current = o

    log("current: "+str(current))

    #move the current tile to the closed set
    openSet.remove(current)
    closedSet.add(current)

    #find the von neumann neighbourhood
    nbd = current.nbd()
Другие вопросы по тегам