题目描述
在一个 2 x 3 的板上(board)有 5 块砖瓦,用数字 1~5 来表示, 以及一块空缺用 0 来表示.
一次移动定义为选择 0 与一个相邻的数字(上下左右)进行交换.
最终当板 board 的结果是 [[1,2,3],[4,5,0]] 谜板被解开。
给出一个谜板的初始状态,返回最少可以通过多少次移动解开谜板,如果不能解开谜板,则返回 -1 。
示例:
输入:board = [[1,2,3],[4,0,5]]
输出:1
解释:交换 0 和 5 ,1 步完成
输入:board = [[1,2,3],[5,4,0]]
输出:-1
解释:没有办法完成谜板
输入:board = [[4,1,2],[5,0,3]]
输出:5
解释:
最少完成谜板的最少移动次数是 5 ,
一种移动路径:
尚未移动: [[4,1,2],[5,0,3]]
移动 1 次: [[4,1,2],[0,5,3]]
移动 2 次: [[0,1,2],[4,5,3]]
移动 3 次: [[1,0,2],[4,5,3]]
移动 4 次: [[1,2,0],[4,5,3]]
移动 5 次: [[1,2,3],[4,5,0]]
输入:board = [[3,2,4],[1,5,0]]
输出:14
解题思路
寻找最短路径的问题可以优先考虑广度优先搜索算法。队列是广度优先搜索的好伴侣。
(1)终止条件判断
通过不断的移动数字‘0’,board上数字满足[[1,2,3],[4,5,0]],返回移动过的次数。
(2)‘0’在board上不同位置时可以移动的方向,或者叫可以跟它换位置的数字所在的位置。
为了便于说明,把board中的二维结构列表变成一维。
board = [[4,1,2],[5,0,3]]
board = board[0] + board[1]
board = [4,1,2,5,0,3]
有6个位置,0~5
在0位置,数字'0'只能与(1,3)位置的数字交换位置。
在1位置,数字'0'只能与(0,2,4)位置的数字交换位置。
在2位置,数字'0'只能与(1,5)位置的数字交换位置。
在3位置,数字'0'只能与(0,4)位置的数字交换位置。
在4位置,数字'0'只能与(2,3,5)位置的数字交换位置。
在5位置,数字'0'只能与(2,4)位置的数字交换位置。
(3)从当前‘0’所在位置开始进行查找,直到满足条件输出最短步长或者尝试所有可能后不满足条件输出-1.
完整代码如下:
class Solution:
def slidingPuzzle(self, board: List[List[int]]) -> int:
board = board[0]+board[1] # 把board连起来变一维
moves = [(1, 3), (0, 2, 4), (1, 5), (0, 4), (1, 3, 5), (2, 4)] # 每个位置的0可以交换的位置
q, visited = [(tuple(board), board.index(0), 0)], set() # bfs队列和已访问状态记录
while q:
state, now, step = q.pop(0) # 分别代表当前状态,0的当前位置和当前步数
if state == (1, 2, 3, 4, 5, 0): # 找到了
return step
for next in moves[now]: # 遍历所有可交换位置
_state = list(state)
_state[next], _state[now] = _state[now], state[next] # 交换位置
_state = tuple(_state)
print(_state)
if _state not in visited: # 确认未访问
q.append((_state, next, step+1))
visited.add(state)
return -1
来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/sliding-puzzle