Puzzle 8 Resolver в Python - я не могу пройти все тесты
Мне нужно сделать решатель 8 головоломок с алгоритмами bfs, dfs и A* в python, но у меня есть некоторые проблемы. Я не могу пройти все тесты. Кто-нибудь может мне помочь?
РЕДАКТИРОВАТЬ: я отредактировал мой код для работы с одномерным массивом. Это работа для:
python driver_3.py bfs 3,1,2,0,4,5,6,7,8
python driver_3.py bfs 1,2,5,3,4,0,6,7,8
но с python driver_3.py bfs 6,1,8,4,0,2,7,3,5
кажется не находит решения. Он работает в течение нескольких минут, не найдя решения.
Это мой код:
driver_3.py
import sys
from EightPuzzle import EightPuzzle as g
if __name__ == "__main__":
method = sys.argv[1]
board = list(map(int,sys.argv[2].split(',')))
print("Start: ", board)
game = g(board)
if (method == 'bfs') | (method == 'dfs') | (method == 'ast') :
game.resolve(alg=method)
else:
print('Method not allowed.')
EightPuzzle.py
from Graph import Graph as Graph
from Node import Node
import time
class EightPuzzle(object):
def __init__(self, initialState):
self.graph = Graph(Node(initialState))
self.actions = ['Up','Down', 'Left', 'Right']
self.goal = [0,1,2,3,4,5,6,7,8]
self.start_time = time.time()
self.end_time = 0
self.diff_time = 0
self.ram = 0
def get_graph(self):
return self.graph
def set_graph(self, other):
self.graph = other
def get_actions(self):
return self.actions
def set_actions(self, other):
self.actions = other
def get_goal(self):
return self.goal
def set_goal(self, other):
self.goal = other
def get_start_time(self):
return self.start_time
def set_start_time(self, other):
self.ram = other
def get_end_time(self):
return self.end_time
def set_end_time(self, other):
self.end_time = other
def set_diff_time(self):
self.diff_time = self.end_time - self.start_time
return None
def get_diff_time(self):
return self.diff_time
def set_ram_usage(self):
import sys
if sys.platform == "win32":
import psutil
self.ram = psutil.Process().memory_info().rss
else:
# import resource
# self.ram = resource.getrusage(resource.RUSAGE_SELF).ru_maxrss
return False
def get_ram(self):
return self.ram
def set_ram(self, other):
self.ram = other
def find_zero(self, node_state):
return node_state.index(0)
def move_up(self, node_state, zero):
up = node_state[:]
val = up[zero-3]
up[zero-3] = 0
up[zero] = val
return up
def move_down(self, node_state, zero):
down = node_state[:]
val = down[zero+3]
down[zero+3] = 0
down[zero] = val
return down
def move_left(self, node_state, zero):
left = node_state[:]
val = left[zero-1]
left[zero-1] = 0
left[zero] = val
return left
def move_right(self, node_state, zero):
right = node_state[:]
val = right[zero+1]
right[zero+1] = 0
right[zero] = val
return right
def find_neighbors(self, node):
node_state = node.get_state()
print("Node: ", node_state)
zero = self.find_zero(node_state)
neighbors = []
if zero > 2:
neighbors.append(Node(self.move_up(node_state, zero), parent=node, action="Up"))
if zero < 6:
neighbors.append(Node(self.move_down(node_state,zero), parent=node, action="Down"))
if (zero != 0) & (zero != 3) & (zero != 6):
neighbors.append(Node(self.move_left(node_state,zero), parent=node, action="Left"))
if (zero != 2) & (zero != 5) & (zero != 8):
neighbors.append(Node(self.move_right(node_state,zero), parent=node, action="Right"))
return neighbors
def success(self):
self.end_time = time.time()
self.set_diff_time()
self.set_ram_usage()
#self.output()
return "Success"
def bfs(self, goal):
print("bfs")
graph = self.get_graph()
while len(graph.get_frontiers()) != 0:
current_node = graph.frontiers.shift()
graph.explored.append(current_node)
if (current_node == goal):
print("Explored ", str(len(graph.get_explored())-1))
print("Solution found!: ", current_node)
return "Done"
neighbors = self.find_neighbors(current_node)
current_node.set_neighbors(neighbors)
for neighbor in neighbors:
if (neighbor not in graph.get_frontiers()) & (neighbor not in graph.get_explored()):
graph.frontiers.append(neighbor)
print("Solution Not Found")
def dfs(self, goal):
print("dfs")
graph = self.get_graph()
while len(graph.get_frontiers()) != 0:
current_node = graph.frontiers.pop()
graph.explored.append(current_node)
if (current_node == goal):
print("Explored ", str(len(graph.get_explored())-1))
print("Solution found!: ", current_node)
return "Done"
neighbors = self.find_neighbors(current_node)
current_node.set_neighbors(neighbors)
for neighbor in neighbors:
if (neighbor not in graph.get_frontiers()) & (neighbor not in graph.get_explored()):
graph.frontiers.prepend(neighbor)
print("Solution Not Found")
def ast(self, goal):
print("ast")
# TO DO
print("Solution Not Found")
# def output(self):
# with open("output.txt", "w") as f:
# f.write("path_to_goal: ", str(self.path))
# f.write("cost_of_path: ", str(len(self.cost_of_path)))
# f.write("node_expanded: ", str(len(self.explored)))
# f.write("max_search_depth: ", str(self.max_depth))
# f.write("running_time: ", str(self.diff_time))
# f.write("max_ram_usage: ", str(self.ram))
# f.closed
# return "Output Done"
def resolve(self, alg):
if alg == 'bfs':
self.bfs(self.goal)
elif alg == 'dfs':
self.dfs(self.goal)
elif alg == 'ast':
self.ast(self.goal)
return None
Node.py
class Node(object):
def __init__(self, state, parent=None, action="Root", path_cost=0):
self.state = state
self.parent = parent
self.action = action
self.path_cost = path_cost
self.depth = 0
self.neighbors = []
if parent:
self.depth = parent.depth + 1
self.path_cost = parent.path_cost + 1
def __eq__(self,other):
return self.state == other
def __str__(self):
return str(self.state)
def __repr__(self):
return str(self.state)
def get_state(self):
return self.state
def set_state(self, new_state):
self.state = new_state
def get_parent(self):
return self.parent
def set_parent(self, new_parent):
self.parent = new_parent
def get_action(self):
return self.action
def set_action(self, other_action):
self.action = other_action
def get_path_cost(self):
return self.path_cost
def set_path_cost(self, other_path):
self.path_cost = other_path
def get_depth(self):
return self.depth
def set_depth(self, other_depth):
self.depth = other_depth
def get_neighbors(self):
return self.neighbors
def set_neighbors(self, other_neighbors):
self.neighbors = other_neighbors
Graph.py
from Frontiers import Frontiers
class Graph:
def __init__(self, initialState):
self.frontiers = Frontiers(initialState)
self.explored = []
self.path = []
self.max_depth = 0
self.cost_of_path = 0
def get_frontiers(self):
return self.frontiers
def set_frontiers(self, other):
self.frontiers = other
def get_explored(self):
return self.explored
def set_explored(self, other):
self.explored = other
def get_path(self):
return self.path
def set_path(self, other):
self.path = other
def get_max_depth(self):
return self.max_depth
def set_max_depth(self, other):
self.max_depth = other
def get_cost_of_path(self):
return self.cost_of_path
def set_cost_of_path(self, other):
self.cost_of_path = other
Frontiers.py
class Frontiers(list):
def __init__(self, initialState):
self.state = [initialState]
def __len__(self):
return len(self.state)
def __contains__(self,other):
return True if other in self.state else False
def __str__(self):
return str(self.state)
def __repr__(self):
return str(self.state)
def set_state(self,new_state):
self.state = new_state
def get_state(self):
return self.state
def append(self, other):
#append element at the end of the frontiers
self.set_state(self.get_state() + [other])
def prepend(self, other):
#append element at the beginning of the frontiers
self.set_state([other] + self.get_state())
def shift(self):
#return first element of frontier state
first = self.get_state()[0]
if(len(self.get_state()) > 1):
self.set_state(self.get_state()[1:])
else:
self.set_state([])
return first
def pop(self):
#return last element of frontier state
last = self.get_state()[-1]
if(len(self.get_state()) > 1):
self.set_state(self.get_state()[:-1])
else:
self.set_state([])
return last
1 ответ
up = node_state[:]
Это только копирует ссылки на списки в node_state
,
> a = [[1,2], [3,4]]
> b = a [:]
> б
[[1, 2], [3, 4]]
> b [0] [0] = 0
> а
[[0, 2], [3, 4]]
Здесь, изменение одного значения во внутреннем списке b
изменяет соответствующее значение в a
также.
Вместо этого сделайте это: up = [list(row) for row in node_state]
> a = [[1,2], [3,4]]
> b = [список (r) для r в a]
> б
[[1, 2], [3, 4]]
> b [0] [0] = 0
> а
[[1, 2], [3, 4]]