add node cost and add lt comparator, adjust a_star and helpers to use node.cost,

This commit is contained in:
matixezor 2021-04-26 13:52:32 +02:00
parent f225f42ba9
commit 4fae45b696
3 changed files with 23 additions and 64 deletions

View File

@ -4,14 +4,18 @@ from pygame import event
class Node: class Node:
def __init__(self, x: int, y: int, direction: int, parent: Node = None, action: event = None): def __init__(self, x: int, y: int, direction: int, parent: Node = None, action: event = None, cost=0):
self.x = x self.x = x
self.y = y self.y = y
self.direction = direction self.direction = direction
self.parent = parent self.parent = parent
self.action = action self.action = action
self.cost = cost
def __eq__(self, other): def __eq__(self, other):
if isinstance(other, Node): if isinstance(other, Node):
return self.x == other.x and self.y == other.y and self.direction == other.direction return self.x == other.x and self.y == other.y and self.direction == other.direction
return False return False
def __lt__(self, other):
return self.cost < other.cost

View File

@ -1,21 +1,19 @@
from typing import List, Tuple
from operator import itemgetter from operator import itemgetter
from typing import List, Tuple
from src.node import Node
from src.tile import Tile
from ap_mine import APMine from ap_mine import APMine
from at_mine import ATMine from at_mine import ATMine
from src.node import Node
from src.tile import Tile
from .helpers import successor, get_path_actions, calculate_priority from .helpers import successor, get_path_actions, calculate_priority
def a_star(field: List[List[Tile]], start_x: int, start_y: int, start_direction: int, goal: Tuple[int, int]): def a_star(field: List[List[Tile]], start_x: int, start_y: int, start_direction: int, goal: Tuple[int, int]):
print(goal)
explored = [] explored = []
node_queue: List[Tuple[int, Node]] = [(field[start_x][start_y].weight, Node(start_x, start_y, start_direction))] node_queue: List[Node] = [Node(start_x, start_y, start_direction)]
# do kolejki dodawać krotke (priorytet, obiekt)
while node_queue: while node_queue:
priority, node = node_queue.pop(0) node = node_queue.pop(0)
# przy get wyciąga element z najniższą wartością priorytetu # przy get wyciąga element z najniższą wartością priorytetu
if (node.x, node.y) == goal: if (node.x, node.y) == goal:
@ -24,14 +22,15 @@ def a_star(field: List[List[Tile]], start_x: int, start_y: int, start_direction:
explored.append(node) explored.append(node)
for x, y, direction, action in successor(field, node.x, node.y, node.direction): for x, y, direction, action in successor(field, node.x, node.y, node.direction):
neighbour_node = Node(x, y, direction, node, action) neighbour_node = Node(x, y, direction, node, action)
neighbour_priority = calculate_priority(field, node, goal) neighbour_node.cost = calculate_priority(field, neighbour_node, goal)
if neighbour_node not in explored and all(neighbour_node not in queue_tuple for queue_tuple in node_queue): if neighbour_node not in explored and neighbour_node not in node_queue:
node_queue.append((neighbour_priority, neighbour_node)) node_queue.append(neighbour_node)
node_queue = sorted(node_queue, key=itemgetter(0)) node_queue.sort()
elif (index := [ elif (index := [
index for index, queue_tuple in enumerate(node_queue) if neighbour_priority > queue_tuple[0] and neighbour_node == queue_tuple[1] index for index, node_in_queue in enumerate(node_queue) if
neighbour_node == node_in_queue and neighbour_node.cost < node_in_queue.cost
]): ]):
node_queue[index[0]] = (neighbour_priority, neighbour_node) node_queue[index[0]] = neighbour_node
node_queue = sorted(node_queue, key=itemgetter(0)) node_queue = sorted(node_queue, key=itemgetter(0))
return False, False return False, False

View File

@ -6,28 +6,6 @@ from src.node import Node
from src.tile import Tile from src.tile import Tile
def successor(field: List[List[Tile]], x: int, y: int, direction: int):
"""
kody klawiszy:
:a -> 97
:d -> 100
:w -> 119
"""
neighbours = []
coord = 'x' if direction in (1, 3) else 'y'
shift = -1 if direction in (0, 3) else 1
neighbours.append((x, y, (direction - 1) % 4, pg.K_a))
neighbours.append((x, y, (direction + 1) % 4, pg.K_d))
if coord == 'x' and 0 <= x + shift <= 9 and field[y][x + shift].number not in (2, 3):
neighbours.append((x + shift, y, direction, pg.K_w))
elif coord == 'y' and 0 <= y + shift <= 9 and field[y + shift][x].number not in (2, 3):
neighbours.append((x, y + shift, direction, pg.K_w))
return neighbours
def successor(field: List[List[Tile]], x: int, y: int, direction: int): def successor(field: List[List[Tile]], x: int, y: int, direction: int):
""" """
kody klawiszy: kody klawiszy:
@ -60,34 +38,12 @@ def get_path_actions(node: Node):
return path[::-1], actions[::-1] return path[::-1], actions[::-1]
def successor_priority(field: List[List[Tile]], x: int, y: int, direction: int):
"""
kody klawiszy:
:a -> 97
:d -> 100
:w -> 119
"""
neighbours = []
coord = 'x' if direction in (1, 3) else 'y'
shift = -1 if direction in (0, 3) else 1
neighbours.append((x, y, (direction - 1) % 4, pg.K_a))
neighbours.append((x, y, (direction + 1) % 4, pg.K_d))
if coord == 'x' and 0 <= x + shift <= 9 and field[y][x + shift].number not in (2, 3):
neighbours.append((x + shift, y, direction, pg.K_w))
elif coord == 'y' and 0 <= y + shift <= 9 and field[y + shift][x].number not in (2, 3):
neighbours.append((x, y + shift, direction, pg.K_w))
return neighbours
def cost(field: List[List[Tile]], node): def cost(field: List[List[Tile]], node):
return field[node.x][node.y].weight return field[node.y][node.x].weight
def calculate_priority(field: List[List[Tile]], node, goaltest: Tuple[int, int]): def calculate_priority(field: List[List[Tile]], node, goal_test: Tuple[int, int]):
return cost(field, node) + heuristic(node, goaltest) return node.parent.cost + cost(field, node) + heuristic(node, goal_test)
def heuristic(a, b): def heuristic(a, b):