「这是我参与11月更文挑战的第 9 天,活动详情查看:2021最后一次更文挑战」。
题目描述
这是 LeetCode 上的 488. 祖玛游戏 ,难度为 困难。
Tag : 「DFS」、「搜索」、「启发式搜索」
你正在参与祖玛游戏的一个变种。
在这个祖玛游戏变体中,桌面上有 一排 彩球,每个球的颜色可能是:红色 'R'
、黄色 'Y'
、蓝色 'B'
、绿色 'G'
或白色 'W'
。你的手中也有一些彩球。
你的目标是 清空 桌面上所有的球。每一回合:
- 从你手上的彩球中选出 任意一颗 ,然后将其插入桌面上那一排球中:两球之间或这一排球的任一端。
- 接着,如果有出现 三个或者三个以上 且 颜色相同 的球相连的话,就把它们移除掉。
- 如果这种移除操作同样导致出现三个或者三个以上且颜色相同的球相连,则可以继续移除这些球,直到不再满足移除条件。
- 如果桌面上所有球都被移除,则认为你赢得本场游戏。
- 重复这个过程,直到你赢了游戏或者手中没有更多的球。
给你一个字符串 board ,表示桌面上最开始的那排球。另给你一个字符串 hand ,表示手里的彩球。请你按上述操作步骤移除掉桌上所有球,计算并返回所需的 最少 球数。如果不能移除桌上所有的球,返回 -1 。
示例 1:
1 | rust复制代码输入:board = "WRRBBW", hand = "RB" |
示例 2:
1 | rust复制代码输入:board = "WWRRBBWW", hand = "WRBRW" |
示例 3:
1 | php复制代码输入:board = "G", hand = "GGGGG" |
示例 4:
1 | rust复制代码输入:board = "RBYYBBRRB", hand = "YRBGB" |
提示:
- 1 <= board.length <= 16
- 1 <= hand.length <= 5
- board 和 hand 由字符
'R'
、'Y'
、'B'
、'G'
和'W'
组成 - 桌面上一开始的球中,不会有三个及三个以上颜色相同且连着的球
搜索 + 剪枝
数据范围 1<=board.length<=161 <= board.length <= 161<=board.length<=16 和 1<=hand.length<=51 <= hand.length <= 51<=hand.length<=5。
为了方便,我们使用 aaa 和 bbb 来代指 boardboardboard 和 handhandhand。
但在爆搜过程中同时维持两个字符串构造会超时,考虑使用一个 int
来记录 handhandhand 的使用情况。
代码:
1 | Java复制代码class Solution { |
- 时间复杂度:略。「爆搜」同时还得考虑「剪枝」的复杂度分析意义不大。
- 空间复杂度:略
AStar 算法
我们建立一个类 Node
来代指当前搜索局面。
1 | Java复制代码class Node { |
显然,直接对此进行 BFS
,会 TLE。
我们考虑将优化 BFS
中使用到的队列改为优先队列:更接近答案的局面先出队进行局面延展。
然后我们考虑如何设计 AStar 的启发式函数。
首先,一个合格的 AStar 启发式函数应当能够确保「估值不会小于理论最小距离」。同时由于启发式的估值函数是针对于最终状态进行估算,因此只确保最终状态的第一次出队时为最短路,其余中间状态的首次出队不一定是最短路,为此我们需要使用哈希表来记录中间状态的距离变化,如果某个局面的最短距离被更新,我们应当将其再次入队。
基于此,我们设计如下的 AStar 的启发式函数:使用哈希表来统计「当前的棋盘 aaa 的彩球数量」&「当前手上拥有的彩球数量」,对「无解情况」和「理论最小次数」进行分析:
- 对于某个彩球 ccc 而言,如果当前棋盘的数量 + 手上的数量 都不足 333 个,那么该局面往下搜索也必然无解,该局面无须入队;
- 对于某个彩球 ccc 而言,如果当前棋盘数量少于 333 个,那么至少需要补充至 333 个才能被消除,而缺少的个数则是「从手上彩球放入棋盘内」的次数,即对于彩球 ccc,我们理论上至少需要消耗 3−cnt3 - cnt3−cnt 次(cntcntcnt 为当前棋盘拥有的彩球 ccc 的数量)。
需要注意的是:对于某个局面 nodenodenode 而言,最终的距离是由「已确定距离」+「估值距离」两部分组成,我们应当根据这两部分之和进行出队,才能确保算法的正确性。
代码:
1 | Java复制代码class Solution { |
- 时间复杂度:略。「爆搜」同时还得考虑「启发式加速」的复杂度分析意义不大。
- 空间复杂度:略
最后
这是我们「刷穿 LeetCode」系列文章的第 No.488
篇,系列开始于 2021/01/01,截止于起始日 LeetCode 上共有 1916 道题目,部分是有锁题,我们将先把所有不带锁的题目刷完。
在这个系列文章里面,除了讲解解题思路以外,还会尽可能给出最为简洁的代码。如果涉及通解还会相应的代码模板。
为了方便各位同学能够电脑上进行调试和提交代码,我建立了相关的仓库:github.com/SharingSour… 。
在仓库地址里,你可以看到系列文章的题解链接、系列文章的相应代码、LeetCode 原题链接和其他优选题解。
本文转载自: 掘金