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’ 组成
- 桌面上一开始的球中,不会有三个及三个以上颜色相同且连着的球
解题思路
使用朴素的回溯法:每次递归尝试向board的每一个位置插入hand里面的每一颗球,插入以后检查是否能出现三个或者三个以上 且 颜色相同 的球相连的话,并且把它们移除掉。
剪枝
- 使用set记录已经递归过的情况
- 一旦当前递归的使用的球数小于了目前得出的最小球数
代码
class Solution {
public:int m=0x3f3f3f3f;void dfs(int cnt,string hand,string board) {if (cnt>=m) return;if (set.count({board,cnt}))return;set.insert({board,cnt});if (board.empty()) {m=min(m,cnt);return;}if (hand.empty()) return;for (int i = 0; i < board.size(); ++i) {for (int j = 0; j < hand.size(); ++j) {char cur=hand[j];hand.erase(hand.begin()+j);string ns=board;ns.insert(ns.begin()+i,cur);handler(ns);dfs(cnt+1,hand,ns);hand.insert(hand.begin()+j,cur);}}}int findMinStep(string board, string hand) {dfs(0,hand, board);return m==0x3f3f3f3f ? -1 : m;}set<pair<string,int>> set ;void handler(string& s){int seq(1);for (int i = 1; i <= s.size(); ++i) {if (i<s.size()&&s[i]==s[i-1]) {seq++;continue;}if(seq>=3){s.erase(i-seq,seq);i=0;}seq=1;}}};