【剪枝】【广度优先】【深度优先】488祖玛游戏

作者推荐

【动态规划】458:可怜的小猪

涉及知识点

剪枝 广度优先 深度优先

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’ 组成
桌面上一开始的球中,不会有三个及三个以上颜色相同且连着的球

剪枝

剪枝一:如果手中有多个相同颜色的球,只需要尝试一个,其它不需要尝试。
剪枝二:如果收中的球和桌面颜色相同的球,只需要考虑插入到最前面。在一个红球的前面或后面插入红球,结果一样两个连续的红球。在两个红球的前面或后面或中间插入红球的效果一样,连续的3个红球消除。
假定插入的颜色是ch,插入的位置是i,只需要考虑以下两种情况:
一,board[i]等于ch。
二,board[i-1]等于board[i]。
忽略剪枝后,再考虑这两种情况。这两种情况都是必须,缺少情况一,无法消除任何球。缺少情况二,无法消除以下用例:
RRWWRRBBRR WB,第一步消除W或B的连锁反应都会消除4个R,造成余下的2个R,无法消除。
RRWWRRBBRWR − − − > 消除 B _{--->}^{消除B} −−−>消除B RRWWRRRWR − − − − − − − − > 连锁消除 3 个 R _{-------->}^{连锁消除3个R} −−−−−−−−>连锁消除3R RRWWWR − − − − − − − − > 连锁消除 3 个 W _{-------->}^{连锁消除3个W} −−−−−−−−>连锁消除3W RRR − − − − − − − − > 连锁消除 3 个 W _{-------->}^{连锁消除3个W} −−−−−−−−>连锁消除3W 结束。

下面来证明为什么只需要考虑两种情况:

0==i两种相等情况一
两种不等假定ch能被消除,假定和ch同时消除,从左向右第一个小标为i1,board[0]和board[i1]一起被消除,说明board(0,i1)能被消除且不依赖不影响board[i1],那将ch不插入到0,插入到i1,操作顺序完全一样。被淘汰。简称证明一。
board.length-1 == i两者相等就是剪枝二。
两者不等类似证明一。
i是中间的下标三者相等剪枝二淘汰。
ch==board[i]就是情况一。
ch==board[i-1]就是剪枝二,淘汰。
board[i-1]==board[i]就是情况二。
三者不等。假定ch能被消除,否则则组合没有任何意义。一个ch无法消除,所以它的左边或边有必定有一个ch被一起消除。不失一般性,假定在它的左边,下标为i1,borad[i1]和ch一起被消除,说明(i1,i)的字符都消除。那将ch插入到i1和i的效果一样。由于board[i-1]不等于board[i]和ch,所以无论是否插入ch,都不会影响右边的字符。 两中方案(i1,i)的左边都是ch,所以新方案不会有影响左边。

第一版代码

class Solution {
public:
int findMinStep(string board, string hand) {
sort(hand.begin(), hand.end());
queue<pair<string, string>> que;
unordered_set setHasDo;
auto Add = [&](const string& s, const string& h)
{
bool bEmpty = s.empty();
const string cur = s + “-” + h;
if (setHasDo.count(cur))
{
return;
}
setHasDo.emplace(cur);
que.emplace(s, h);
};
Add(board, hand);
while (que.size())
{
auto [s, h] = que.front();
que.pop();
if (s.empty())
{
return hand.length() - h.length();
}
unordered_map<char, int> mCharCount;
for (const auto& ch : h)
{
mCharCount[ch]++;
}
for (int j = 0; j < s.length(); j++)
{
if ((j + 1 < s.length()) && (s[j] == s[j + 1]))
{
if (mCharCount.count(s[j]) && (mCharCount[s[j]] >= 1))
{
mCharCount[s[j]]–;
Add(Do(s.substr(0,j+1)+s[j] + s.substr(j+1)),ToString(mCharCount));
mCharCount[s[j]]++;
}
for ( auto& [ch, iCnt] : mCharCount)
{
if (ch != s[j])
{
iCnt–;
Add(Do(s.substr(0, j+1) + ch + s.substr(j + 1)), ToString(mCharCount));
iCnt++;
}
}
}
else
{
if (mCharCount.count(s[j]) && (mCharCount[s[j]] >= 2))
{
mCharCount[s[j]] -= 2;
Add(Do(s.substr(0, j) + s[j] + s[j] + s.substr(j)), ToString(mCharCount));
mCharCount[s[j]] += 2;
}
}
}
}
return -1;
}
string Do(const string& s)
{
stack<pair<char, int>> sta;
for (const char& ch : s )
{
while (sta.size() && (sta.top().first != ch) && (sta.top().second >= 3))
{
sta.pop();
}
if (sta.size() && (sta.top().first == ch))
{
sta.top().second++;
}
else
{
sta.emplace(ch, 1);
}
}
string sRet;
while (sta.size())
{
const auto [ch, iCnt] = sta.top();
sta.pop();
if (iCnt < 3)
{//最后一个元素需要判断
sRet += string(iCnt, ch);
}
}
return sRet; //正序和反序是一样的
}
string ToString(const unordered_map<char, int>& mCharCount)
{
string strRet;
for (const auto& [ch, iCnt] : mCharCount)
{
strRet += string(iCnt, ch);
}
return strRet;
}
}; ,

测试用例

template<class T>
void Assert(const T& t1, const T& t2)
{assert(t1 == t2);
}template<class T>
void Assert(const vector<T>& v1, const vector<T>& v2)
{if (v1.size() != v2.size()){assert(false);return;}for (int i = 0; i < v1.size(); i++){Assert(v1[i], v2[i]);}
}int main()
{string board,  hand;	{Solution sln;board = "RRWWRRBBRR", hand = "WB";auto res = sln.findMinStep(board, hand);Assert(2, res);}{Solution sln;board = "WWRRBBWW", hand = "WRBRW";auto res = sln.findMinStep(board, hand);Assert(2, res);}{Solution sln;board = "G", hand = "GGGGG";auto res = sln.findMinStep(board, hand);Assert(2, res);}{Solution sln;board = "WRRBBW", hand = "RB";auto res = sln.findMinStep(board, hand);Assert(-1, res);}{Solution sln;board = "WRRBBW", hand = "RB";auto res = sln.findMinStep(board, hand);Assert(-1, res);}{Solution sln;board = "RBYYBBRRB", hand = "YRBGB";auto res = sln.findMinStep(board, hand);Assert(3, res);}{Solution sln;board = "RRGGBBYYWWRRGGBB", hand = "RGBYW";auto res = sln.findMinStep(board, hand);Assert(-1, res);}}

小的优化

class Solution {
public:int findMinStep(string board, string hand) {sort(hand.begin(), hand.end());queue<pair<string,string>> que;unordered_set<string> setHasDo;auto Add = [&](const string& s, const string& h){const string cur = s + "-" + h;if (setHasDo.count(cur)){return;}setHasDo.emplace(cur);que.emplace(s,h);};Add(board, hand);while (que.size()){auto[s,h] = que.front();que.pop();	if (s.empty()){return hand.length() - h.length();}for (int i = 0 ; i < h.length();i++){if (i && (h[i] == h[i - 1])){//剪枝一continue;}string h2 = h.substr(0,i) + h.substr(i+1);for (int j = 0; j < s.length(); j++){if (j&& (h[i] == s[j-1])){continue;//剪枝二}if ((h[i] == s[j])||(j && (s[j - 1] == s[j]))){Add(Do(s, j, h[i]), h2);}}}}return -1;}string Do(const string& s, int l1,  const char& ch){		stack<pair<char,int>> sta;auto Add = [&sta](const char& ch){while (sta.size() && (sta.top().first != ch ) && (sta.top().second >= 3)){sta.pop();}if (sta.size()&& (sta.top().first == ch)){sta.top().second++;}else{sta.emplace(ch,1);}			};for (int i = 0; i < l1; i++){Add( s[i]);}Add(ch);for (int i = l1; i < s.length() ; i++){Add(s[i]);}string sRet;while (sta.size()){const auto [ch,iCnt] = sta.top();sta.pop();if (iCnt < 3){//最后一个元素需要判断sRet += string(iCnt, ch);}}return sRet; //正序和反序是一样的}
};

实际上只需要考虑三种情况

  • 弹出一个球,三消。
  • 弹出两个同颜色的球三消。
  • 两个同颜色的球直接插入不同颜色的球。
class Solution {
public:int findMinStep(string board, string hand) {sort(hand.begin(), hand.end());queue<pair<string, string>> que;unordered_set<string> setHasDo;auto Add = [&](const string& s, const string& h){bool bEmpty = s.empty();const string cur = s + "-" + h;if (setHasDo.count(cur)){return;}setHasDo.emplace(cur);que.emplace(s, h);};Add(board, hand);while (que.size()){auto [s, h] = que.front();que.pop();if (s.empty()){return hand.length() - h.length();}unordered_map<char, int> mCharCount;for (const auto& ch : h){mCharCount[ch]++;}			for (int j = 0; j < s.length(); j++){if ((j + 1 < s.length()) && (s[j] == s[j + 1])){if (mCharCount.count(s[j]) && (mCharCount[s[j]] >= 1)){mCharCount[s[j]]--;Add(Do(s.substr(0,j+1)+s[j] + s.substr(j+1)),ToString(mCharCount));mCharCount[s[j]]++;}for ( auto& [ch, iCnt] : mCharCount){if (ch != s[j]){iCnt--;Add(Do(s.substr(0, j+1) + ch + s.substr(j + 1)), ToString(mCharCount));iCnt++;}}}else{if (mCharCount.count(s[j]) && (mCharCount[s[j]] >= 2)){mCharCount[s[j]] -= 2;Add(Do(s.substr(0, j) + s[j] + s[j] + s.substr(j)), ToString(mCharCount));mCharCount[s[j]] += 2;}}				}}return -1;}string Do(const string& s){stack<pair<char, int>> sta;for (const char& ch : s ){while (sta.size() && (sta.top().first != ch) && (sta.top().second >= 3)){sta.pop();}if (sta.size() && (sta.top().first == ch)){sta.top().second++;}else{sta.emplace(ch, 1);}}string sRet;while (sta.size()){const auto [ch, iCnt] = sta.top();sta.pop();if (iCnt < 3){//最后一个元素需要判断sRet += string(iCnt, ch);}}return sRet; //正序和反序是一样的}string ToString(const unordered_map<char, int>& mCharCount){string strRet;for (const auto& [ch, iCnt] : mCharCount){strRet += string(iCnt, ch);}return strRet;}
};

进一步改进

hand的状态可以用位运算来表示,这可以提速,本题已经够复杂了。就不继续了。
可以用 深度优先,但要处理重复搜索。
已有状态: board用字符串,字典树提速。 hand用int 提速。

2023年1月版

class Solution {
public:
int findMinStep(string board, string hand) {
std::sort(hand.begin(), hand.end());
m_iMaxStep = hand.size();
std::unordered_set pre;
pre.emplace(hand + board);
for (int iStep = m_iMaxStep; iStep > 0; iStep–)
{
std::unordered_set dp;
for (const auto str : pre)
{
const string sHand = str.substr(0, iStep);
const string sBoard = str.substr(iStep);
if (“” == sBoard)
{
return m_iMaxStep - iStep;
}
Do(dp, sBoard, sHand);
}
pre.swap(dp);
}
if (pre.count(“”))
{
return m_iMaxStep;
}
return -1;
}
void Do(std::unordered_set& dp, const string& strBoard, const string& strHand)
{
for (int i = 0; i < strHand.size(); i++)
{
if ((i>0) && (strHand[i] == strHand[i - 1]))
{
continue;
}
string tmp = strHand;
tmp.erase(i, 1);
const char& ch = strHand[i];
for (int j = 0; j <= strBoard.size(); j++)
{
bool bNeedDo = false;
if (j < strBoard.length() && (strBoard[j] == ch))
{
bNeedDo = true;
}
if (j>0)
{
if ((j < strBoard.length()) && (strBoard[j-1] == strBoard[j])&&(ch != strBoard[j]))
{
bNeedDo = true;
}
}
if (!bNeedDo)
{
continue;
}
dp.emplace(tmp + Do(strBoard, j, ch));
}
}
}
string Do(string strBoard, int index, const char& ch)
{
strBoard.insert(index, std::string(“”) + ch);
Erase(strBoard, index);
return strBoard;
}
void Erase(string& strBoard, int index)
{
string res;
vector<pair<char, int>> st;
for (auto c : strBoard) {
while (!st.empty() && c != st.back().first && st.back().second >= 3) {
st.pop_back();
}
if (st.empty() || c != st.back().first) {
st.push_back({ c, 1 });
}
else {
st.back().second++;
}
}
if (!st.empty() && st.back().second >= 3) {
st.pop_back();
}
for (int i = 0; i < st.size(); ++i) {
for (int j = 0; j < st[i].second; ++j) {
res.push_back(st[i].first);
}
}
strBoard = res;
}
int m_iMaxStep;
};

扩展阅读

视频课程

有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
https://edu.csdn.net/course/detail/38771

如何你想快

速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176

相关

下载

想高屋建瓴的学习算法,请下载《喜缺全书算法册》doc版
https://download.csdn.net/download/he_zhidan/88348653

我想对大家说的话
闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。
子墨子言之:事无终始,无务多业。也就是我们常说的专业的人做专业的事。
如果程序是一条龙,那算法就是他的是睛

测试环境

操作系统:win7 开发环境: VS2019 C++17
或者 操作系统:win10 开发环境: VS2022 **C+

+17**
如无特殊说明,本算法用**C++**实现。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/617717.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

kubeadm安装kubernetes

基本环境配置 节点分为&#xff1a;master&#xff0c;node&#xff0c;masterlb(keepalived虚拟Ip&#xff0c;不占用机器) k8s-master01 16 k8s-node01 113 15 k8s-node02 115 进入之后直接选done done 上海 123456 设置静态ip 然后去虚拟机里面设置ens即可 查看命…

MFC为对话框资源添加类

VC6新建一个对话框类型的工程; 建立之后资源中默认有2个对话框,一个是主对话框,About这个是默认建立的关于版权信息的; 然后主对话框有对应的.h和.cpp文件;可以在其中进行编程; 默认建立的有一个 关于 对话框; 在资源中新插入一个对话框,IDD_DIALOG1是对话框ID; 新加…

连接数问题

连接数问题 在使用mysql时有时候会报too many connections错误&#xff0c;这是连接数过多导致的&#xff0c;然后就会去修改max_connections参数&#xff0c;但是这个参数也不能无上限的增大&#xff0c;容易造成机器内存不足&#xff0c;还是需要找到为什么会有这么多连接的原…

国际版WPS Office 18.6.1

【应用名称】&#xff1a;WPS Office 【适用平台】&#xff1a;#Android 【软件标签】&#xff1a;#WPS 【应用版本】&#xff1a;18.6.1 【应用大小】&#xff1a;160MB 【软件说明】&#xff1a;软件日常更新。WPS Office是使用人数最多的移动办公软件。独有手机阅读模式…

LLM主流框架:Causal Decoder、Prefix Decoder和Encoder-Decoder

本文将介绍如下内容&#xff1a; transformer中的mask机制Causal DecoderPrefix DecoderEncoder Decoder总结 一、transformer中的mask机制 在Transformer模型中&#xff0c;mask机制是一种用于在self-attention中的技术&#xff0c;用以控制不同token之间的注意力交互。具体…

【Redis】Redis初识

一、redis概述 1、redis redis&#xff1a;一款高性能的NOSQL系列的非关系型数据库redis 2、NoSQL NoSQL&#xff1a;泛指非关系型的数据库 NOSQL分类&#xff1a; 键值(Key-Value)存储数据库&#xff1a;内容缓存&#xff0c;主要用于处理大量数据的高访问负载列存储数据…

VUE 前端框架学习总结

Vue 的核心库只关注视图层&#xff0c;方便与第三方库或既有项目整合。 视图层即页面展示&#xff08;给用户看的部分&#xff09;&#xff0c;刷新后台给的数据&#xff1a;HTML CSS JS Vue 只负责视图层&#xff0c;其他前端部分有以下技术解决 网络通信 &#xff1a; axi…

探索WPF控件内容模型的四大支柱

WPF 内容模型 WPF控件内容模型主要指派生于System.Windows.Controls.Control类的各种控件&#xff0c;有四个可包含任意内容的类。 下表列出了继承自 Control 的类。 ContentControl&#xff1a;用于包含一段任意类型的内容。但是只能包含一个子元素作为其“内容”。它可以包…

arm服务器和麒麟v10安装nacos

在arm飞腾服务器和麒麟V10SP3上安装nacos 服务器和系统版本 ############## Kylin Linux Version ################# Release: Kylin Linux Advanced Server release V10 (Lance)Kernel: 4.19.90-52.22.v2207.ky10.aarch64Build: Kylin Linux Advanced Server release V10 (S…

图像分割deeplab系列

DeepLab系列是谷歌团队提出的一系列语义分割算法。DeepLab v1于2014年推出&#xff0c;并在PASCAL VOC2012数据集上取得了分割任务第二名的成绩&#xff0c;随后2017到2018年又相继推出了DeepLab v2&#xff0c;DeepLab v3以及DeepLab v3。DeepLab v1的两个创新点是空洞卷积&am…

【Python】新鲜出炉的海洋捕食者算法Python版本

2020年发表的海洋捕食者算法《Marine Predators Algorithm: A nature-inspired metaheuristic》。 作者只在原论文中给出了MATLAB代码&#xff0c;网上也没有Python版本&#xff0c;我自己用Python重写了MATLAB代码。 """2020海洋捕食者算法 """…

spaceship

通过数字平台启动您的网站、想法和未来&#xff0c;该平台旨在提供和连接您所需的域、托管、电子邮件和 Web 工具&#xff0c;并让您完全掌控 如果需要购买可以开5347的卡&#xff0c;点击获取

python 多线程 简介

python多线程简介 多线程的概念是相对单线程而言的。所谓单线程是指CPU在处理完成一项任务之前是不会开始处理第二件任务的。简单来说&#xff0c;单线程在执行任务时是有一定的顺序的。而随着科技的进步&#xff0c;CPU等计算机组件的升级换代日新月异&#xff0c;CPU处理速度…

小学信息科技Python课程第2课:坐标与画笔

一、turtle画布与坐标系 在同一平面互相垂直且有公共原点的两条数轴构成平面直角坐标系。在坐标系中&#xff0c;水平方向的轴都称为x轴&#xff0c;垂直方向的轴都称为y轴 它们相交于O点&#xff0c;在这一个点里&#xff0c;x轴的值为0&#xff0c;y轴的值也为0&#xff0c;所…

掌握 Vue 响应式系统,让数据驱动视图(下)

&#x1f90d; 前端开发工程师&#xff08;主业&#xff09;、技术博主&#xff08;副业&#xff09;、已过CET6 &#x1f368; 阿珊和她的猫_CSDN个人主页 &#x1f560; 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 &#x1f35a; 蓝桥云课签约作者、已在蓝桥云…

SpringBoot全局配置Long转String丢失精度的问题解决

第一种方式 简单粗暴&#xff0c;将所有的Long类型&#xff0c;改为String&#xff0c;数据库改成varchar类型&#xff1b; 第二种方式 自己建个配置类 extends WebMvcConfigurerAdapter 已经被弃用&#xff0c;直接实现WebMvcConfigurer该接口就行了 EnableWebMvc Config…

什么是网络数据抓取?有什么好用的数据抓取工具?

一、什么是网络数据抓取 网络数据抓取&#xff08;Web Scraping&#xff09;是指采用技术手段从大量网页中提取结构化和非结构化信息&#xff0c;按照一定规则和筛选标准进行数据处理&#xff0c;并保存到结构化数据库中的过程。目前网络数据抓取采用的技术主要是对垂直搜索引…

DNS解析和它的三个实验

一、DNS介绍 DNS&#xff1a;domain name server 7层协议 名称解析协议 tcp /53 主从之间的同步 udp/53 名字解析 DNS作用&#xff1a;将域名转换成IP地址的协议 1.1DNS的两种实现方式 1.通过hosts文件&#xff08;优先级最高&#xff09; 分散的管理 linux /etc/hos…

深度生成模型(Deep Generative Models)

什么是机器学习 深度生成模型&#xff08;Deep Generative Models&#xff09;是一类利用深度学习方法生成新样本的模型。这些模型通常被用于生成与训练数据集相似的新数据&#xff0c;例如图像、文本或音频。深度生成模型的两个主要类型是生成对抗网络&#xff08;GANs&#…

QA面试题

1、质量保证(QA)是什么&#xff1f; QA代表质量保证。QA 是一组活动&#xff0c;旨在确保开发的软件满足 SRS 文档中提到的所有规范或要求。QA 遵循 PDCA 循环&#xff1a; 计划/Plan - 计划是质量保证的一个阶段&#xff0c;组织在此阶段确定构建高质量软件产品所需的过程。做…