488. 祖玛游戏
难度 困难
你正在参与祖玛游戏的一个变种。
在这个祖玛游戏变体中,桌面上有 一排 彩球,每个球的颜色可能是:红色 'R'
、黄色 'Y'
、蓝色 ‘B’、绿色 'G'
或白色 'W'
。你的手中也有一些彩球。
你的目标是 清空 桌面上所有的球。每一回合:
- 从你手上的彩球中选出 任意一颗 ,然后将其插入桌面上那一排球中:两球之间或这一排球的任一端。
- 接着,如果有出现 三个或者三个以上 且 颜色相同 的球相连的话,就把它们移除掉。
- 如果这种移除操作同样导致出现三个或者三个以上且颜色相同的球相连,则可以继续移除这些球,直到不再满足移除条件。
- 如果桌面上所有球都被移除,则认为你赢得本场游戏。
- 重复这个过程,直到你赢了游戏或者手中没有更多的球。‘
给你一个字符串 board
,表示桌面上最开始的那排球。另给你一个字符串 hand
,表示手里的彩球。请你按上述操作步骤移除掉桌上所有球,计算并返回所需的 最少 球数。如果不能移除桌上所有的球,返回 -1
。
示例 1:
输入:board = "WRRBBW", hand = "RB" 输出:-1 解释:无法移除桌面上的所有球。可以得到的最好局面是: - 插入一个 'R' ,使桌面变为 WRRRBBW 。WRRRBBW -> WBBW - 插入一个 'B' ,使桌面变为 WBBBW 。WBBBW -> WW 桌面上还剩着球,没有其他球可以插入。
示例 2:
输入:board = "WWRRBBWW", hand = "WRBRW" 输出:2 解释:要想清空桌面上的球,可以按下述步骤: - 插入一个 'R' ,使桌面变为 WWRRRBBWW 。WWRRRBBWW -> WWBBWW - 插入一个 'B' ,使桌面变为 WWBBBWW 。WWBBBWW -> WWWW -> empty 只需从手中出 2 个球就可以清空桌面。
示例 3:
输入:board = "G", hand = "GGGGG" 输出:2 解释:要想清空桌面上的球,可以按下述步骤: - 插入一个 'G' ,使桌面变为 GG 。 - 插入一个 'G' ,使桌面变为 GGG 。GGG -> empty 只需从手中出 2 个球就可以清空桌面。
示例 4:
输入:board = "RBYYBBRRB", hand = "YRBGB" 输出:3 解释:要想清空桌面上的球,可以按下述步骤: - 插入一个 'Y' ,使桌面变为 RBYYYBBRRB 。RBYYYBBRRB -> RBBBRRB -> RRRB -> B - 插入一个 'B' ,使桌面变为 BB 。 - 插入一个 'B' ,使桌面变为 BBB 。BBB -> empty 只需从手中出 3 个球就可以清空桌面。
提示:
1 <= board.length <= 16
1 <= hand.length <= 5
board
和hand
由字符'R'
、'Y'
、'B'
、'G'
和'W'
组成- 桌面上一开始的球中,不会有三个及三个以上颜色相同且连着的球
题解
官方
广度优先搜索
class Solution: def findMinStep(self, board: str, hand: str) -> int: def clean(s): # 消除桌面上需要消除的球 n = 1 while n: s, n = re.subn(r"(.)\1{2,}", "", s) return s hand = "".join(sorted(hand)) # 初始化用队列维护的状态队列:其中的三个元素分别为桌面球状态、手中球状态和回合数 queue = deque([(board, hand, 0)]) # 初始化用哈希集合维护的已访问过的状态 visited = {(board, hand)} while queue: cur_board, cur_hand, step = queue.popleft() for i, j in product(range(len(cur_board) + 1), range(len(cur_hand))): # 第 1 个剪枝条件: 当前球的颜色和上一个球的颜色相同 if j > 0 and cur_hand[j] == cur_hand[j - 1]: continue # 第 2 个剪枝条件: 只在连续相同颜色的球的开头位置插入新球 if i > 0 and cur_board[i - 1] == cur_hand[j]: continue # 第 3 个剪枝条件: 只在以下两种情况放置新球 # - 第 1 种情况 : 当前球颜色与后面的球的颜色相同 # - 第 2 种情况 : 当前后颜色相同且与当前颜色不同时候放置球 choose = False if 0 < i < len(cur_board) and cur_board[i - 1] == cur_board[i] and cur_board[i - 1] != cur_hand[j]: choose = True if i < len(cur_board) and cur_board[i] == cur_hand[j]: choose = True if choose: new_board = clean(cur_board[:i] + cur_hand[j] + cur_board[i:]) new_hand = cur_hand[:j] + cur_hand[j + 1:] if not new_board: return step + 1 if (new_board, new_hand) not in visited: queue.append((new_board, new_hand, step + 1)) visited.add((new_board, new_hand)) return -1 作者:LeetCode-Solution 链接:https://leetcode-cn.com/problems/zuma-game/solution/zu-ma-you-xi-by-leetcode-solution-lrp4/ 来源:力扣(LeetCode) 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
深度优先搜索
import re from functools import lru_cache from itertools import product class Solution: def findMinStep(self, board: str, hand: str) -> int: ans = self.dfs(board, "".join(sorted(hand))) return ans if ans <= 5 else -1 @lru_cache(None) def dfs(self, cur_board: str, cur_hand: str): if not cur_board: return 0 res = 6 for i, j in product(range(len(cur_board) + 1), range(len(cur_hand))): # 第 1 个剪枝条件: 手中颜色相同的球只需要考虑其中一个即可 if j > 0 and cur_hand[j] == cur_hand[j - 1]: continue # 第 2 个剪枝条件: 只在连续相同颜色的球的开头位置插入新球 if i > 0 and cur_board[i - 1] == cur_hand[j]: continue # 第 3 个剪枝条件: 只考虑放置新球后有可能得到更优解的位置 # - 第 1 种情况 : 当前球颜色与后面的球的颜色相同 # - 第 2 种情况 : 当前后颜色相同且与当前颜色不同时候放置球 choose = False if 0 < i < len(cur_board) and cur_board[i - 1] == cur_board[i] and cur_board[i - 1] != cur_hand[j]: choose = True if i < len(cur_board) and cur_board[i] == cur_hand[j]: choose = True if choose: new_board = self.clean(cur_board[:i] + cur_hand[j] + cur_board[i:]) new_hand = cur_hand[:j] + cur_hand[j + 1:] res = min(res, self.dfs(new_board, new_hand) + 1) return res @staticmethod def clean(s): n = 1 while n: s, n = re.subn(r'(.)\1{2,}', '', s) return s 作者:LeetCode-Solution 链接:https://leetcode-cn.com/problems/zuma-game/solution/zu-ma-you-xi-by-leetcode-solution-lrp4/ 来源:力扣(LeetCode) 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。