IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 力扣每日一题2021-11-09祖玛游戏 -> 正文阅读

[数据结构与算法]力扣每日一题2021-11-09祖玛游戏


488.祖玛游戏

题目描述

祖玛游戏


思路:BFS、DFS

根据题意,桌面上最多有16个球,手中最多5个球;可以按照任意顺序在5个回合内使用手中的球;在每个回合中,我们可以选择将手中的球插入到桌面上任意两球之间或者这一排球的任意一端。

BFS

根据上述内容,可以通过如下方法实现BFS:
使用队列维护需要处理的状态队列,使用哈希集合存储已经访问过的状态。每次取出队列队头的状态,考虑其中所有可以插入球的方案,如果新方案还没有被访问过,则将新方案添加到队列的队尾。

剪枝条件:

  • 手中的颜色相同的球每次选择时,考虑其中一个即可。这是因为球颜色一样,在一次放入球的操作中,选择相同颜色球中的任意一个对结果没有影响。在具体实现中,可以对手中的球的颜色进行排序,如果当前遍历的球的颜色和上一个遍历的球的颜色相同,则跳过当前遍历的球。当然也可以Python也可以使用Counter方法统计球的颜色个数。
  • 只在连续相同的颜色的球的开头位置或者结尾位置插入新的颜色相同的球。假设桌上有一个白球,那么在左侧和右侧加入一个新的白球没有区别;假定桌上有两个连续的白球,则在其左侧、右侧或中间插入一个新的白球也没有区别。因此,如果新插入的球和桌面上某组连续颜色相同的球(也可以是1个球)的颜色相同,则只需要考虑在其左侧或右侧中的一种情况插入新球即可。
  • 只考虑放置新球后可能得到的更优解的位置。考虑插入新球的颜色与插入位置周围球的颜色的情况,再根据第二个剪枝条件剪枝后,还可能出现如下三种情况:插入新球与插入位置右侧的球的颜色相同;插入新球与插入位置两侧的球颜色均不相同,且插入位置两侧的球的颜色不同;插入新球与插入位置两侧的球颜色均不相同,且插入位置两侧的球的颜色相同。

因为题目规定了如果在消除后,出现了新的连续的三个或三个以上颜色相同的球,则继续消除这些球,直到不再满足消除条件,实际操作时,可以利用栈的特性,每次遇到可消除的球,就将其从栈中弹出。可以在遍历桌上的球时,使用列表维护遍历过的每种球的颜色和连续数量,从而通过一次遍历消除连续三个或三个以上颜色相同的球。

Python实现

Python实现

# BFS
class Solution:
    def findMinStep(self, board: str, hand: str) -> int:
        def clean(s):
            # 消除桌面上需要消除的球
            n = 1
            while n:
                s, n = re.subn(r"(.)\1{2,}", "", s)
            return s
        
        hand = "".join(sorted(hand))
        
        # 初始化用队列维护的状态队列,其中的三个元素分别为桌面球状态、手中球状态和回合数
        queue = deque([(board, hand, 0)])
        
        # 初始化用哈希集合维护的已访问过的状态
        visited = {(board, hand)}
        
        while queue:
            cur_board, cur_hand, step = queue.popleft()
            for i, j in product(range(len(cur_board)+1), range(len(cur_hand))):
                # 当前球的颜色与上一个球的颜色相同
                if j > 0 and cur_hand[j] == cur_hand[j-1]:
                    continue
                # 只在连续相同颜色的球的开头位置插入新球
                if i > 0 and cur_board[i-1] == cur_hand[j]:
                    continue
                # 当前球颜色与后面的球的颜色相同或当前后颜色相同且与当前颜色不同时放置球
                choose = False
                if 0 < i < len(cur_board) and cur_board[i - 1] == cur_board[i] and cur_board[i - 1] != cur_hand[j]:
                    choose = True
                if i < len(cur_board) and cur_board[i] == cur_hand[j]:
                    choose = True
                if choose:
                    new_board = clean(cur_board[:i]+cur_hand[j]+cur_board[i:])
                    new_hand = cur_hand[:j] + cur_hand[j+1:]
                    if not new_board:
                        return step + 1
                    if (new_board, new_hand) not in visited:
                        queue.append((new_board, new_hand, step+1))
                        visited.add((new_board, new_hand))
        return -1

DFS

明天补上

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-11-10 12:39:06  更:2021-11-10 12:41:05 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/26 10:24:24-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码