祖玛游戏
难度:
标签:
题目描述
你正在参与祖玛游戏的一个变种。
在这个祖玛游戏变体中,桌面上有 一排 彩球,每个球的颜色可能是:红色 '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'
组成- 桌面上一开始的球中,不会有三个及三个以上颜色相同且连着的球
代码结果
运行时间: 632 ms, 内存: 26.1 MB
/*
思路:
1. 使用Java Streams API简化操作,但本题使用Streams优势不大,主要体现在遍历和集合操作上。
2. 仍然是回溯法,尝试所有可能的插入方案。
3. 对于每一种方案,计算出插入后形成三个或以上相同颜色的球后,消除的结果。
4. 使用一个哈希表记录已经处理过的局面,避免重复计算。
*/
import java.util.HashMap;
import java.util.Map;
import java.util.stream.IntStream;
public class ZumaGameStream {
public int findMinStep(String board, String hand) {
Map<String, Integer> memo = new HashMap<>();
int result = dfs(board, hand, memo);
return result == Integer.MAX_VALUE ? -1 : result;
}
private int dfs(String board, String hand, Map<String, Integer> memo) {
if (board.isEmpty()) return 0;
if (hand.isEmpty()) return Integer.MAX_VALUE;
String key = board + " " + hand;
if (memo.containsKey(key)) return memo.get(key);
int result = IntStream.range(0, hand.length())
.mapToObj(i -> hand.charAt(i))
.flatMap(c -> IntStream.rangeClosed(0, board.length())
.mapToObj(j -> insertAndEliminate(board, j, c, hand.substring(0, hand.indexOf(c)) + hand.substring(hand.indexOf(c) + 1))))
.filter(newBoard -> newBoard != null)
.map(newBoard -> dfs(newBoard, hand.replaceFirst(c + "", ""), memo) + 1)
.min(Integer::compareTo)
.orElse(Integer.MAX_VALUE);
memo.put(key, result);
return result;
}
private String insertAndEliminate(String board, int pos, char c, String newHand) {
StringBuilder newBoard = new StringBuilder(board);
newBoard.insert(pos, c);
return eliminate(newBoard.toString());
}
private String eliminate(String board) {
int i = 0;
while (i < board.length()) {
int j = i;
while (j < board.length() && board.charAt(j) == board.charAt(i)) j++;
if (j - i >= 3) {
return eliminate(board.substring(0, i) + board.substring(j));
} else {
i = j;
}
}
return board;
}
}
解释
方法:
该题解采用深度优先搜索(DFS)的思路。首先将手中的球按字典序排序,然后对棋盘进行 DFS。在 DFS 过程中,尝试把手中的球插入到棋盘的所有可能位置,然后进行消除操作。如果最终棋盘被清空,则返回使用的球数;否则返回无穷大。为了避免重复计算,使用记忆化搜索来缓存中间结果。
时间复杂度:
O(M * N * (N + M^N))
空间复杂度:
O(M * N + M^N * N)
代码细节讲解
🦆
为什么需要将手中的球按字典序排序?这一操作对算法的效果有何影响?
▷🦆
在深度优先搜索中,为什么选择跳过连续相同颜色的球和连续相同颜色的插入位置?这样的处理对算法的效率和结果有什么具体影响?
▷🦆
在进行球的消除操作时,你提到了‘向两边扩展,找到最大的消除范围’,这一策略是如何确保每次操作都能达到最优消除效果的?
▷🦆
递归搜索时,为什么选择在找到一个有效的消除操作后仍然继续尝试其他可能的插入和消除组合?
▷