BFS - 祖玛游戏 - LeetCode 488

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


BFS 爆搜所有状态。

状态数粗略估算:

记   b o a r d   的 长 度 为   L b ,   h a n d   的 长 度 为   L h 记\ board\ 的长度为\ L_b,\ hand\ 的长度为\ L_h  board  Lb hand  Lh

①、任意的插入顺序: L h   的 全 排 列 = L h   ! L_h\ 的全排列=L_h\ ! Lh =Lh !

②、插入的任意位置约有: L b L h L_b^{L_h} LbLh 种可能

故状态总数的上限约为:

≈ 5 ! × 1 6 5 = 120 × 1 0 6 = 1.2 × 1 0 8 ≈ 5!×16^5=120×10^6=1.2×10^8 5!×165=120×106=1.2×108

注意使用 string 的方法。

insert 方法:std::string::insert
erase 方法:std::string::erase

注意多测,要清空哈希表。

代码:

#define x first
#define y second
typedef pair<string, string> status;
unordered_map<string, int> dis;

class Solution {
public:
    void clc(string& str)
    {
        int i = 0, j = 0;
        while(i <= j && j < str.size())
        {
            while(str[j] == str[i] && j < str.size()) j ++;
            if(j - i > 2)
            {
                str.erase(i, j - i);
                i = max(i - 2, 0);	// 删除后,可能 i 之前的两个字符和第 j 个字符相同
                j = i;
            }
            else i = j;
        }
    }

    int findMinStep(string board, string hand) 
    {
        dis.clear();
        
        queue<status> Q;
        status start = {board, hand};
        Q.push(start);
        dis[start.x] = 0;
        
        while(Q.size()) 
        {
            status t = Q.front();
            Q.pop();
            
            if(t.x.empty()) return dis[t.x];
            string u = t.x, v = t.y;
            for(int i = 0; i < v.size(); i ++)
            {
                char c = v[i];
                for(int j = 0; j <= u.size(); j ++)
                {
                    string s = u;
                    string p = v;
                    s.insert(j, 1, c);
                    clc(s);
                    if(!dis.count(s)) 
                    {
                        dis[s] = dis[u] + 1;
                        Q.push({s, p.erase(i, 1)});
                    }
                }
            }
        }
        
        return -1;
    }
};
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值