当前位置: 首页 > news >正文

【C/C++】滑动谜题(leetcode T773)

核心考点:广度优先搜索 (BFS)、哈希表、字符串、状态转移

题目描述:

在一个 2 x 3 的板上(board)有 5 块砖瓦,用数字 1~5 来表示, 以及一块空缺用 0 来表示。一次 移动 定义为选择 0 与一个相邻的数字(上下左右)进行交换.

最终当板 board 的结果是 [[1,2,3],[4,5,0]] 谜板被解开。

给出一个谜板的初始状态 board ,返回最少可以通过多少次移动解开谜板,如果不能解开谜板,则返回 -1 。

重点通过测试用例来理解题目:

示例 1:

输入:board = [[1,2,3],[4,0,5]]
输出:1
解释:交换 0 和 5 ,1 步完成

示例 2:

输入:board = [[1,2,3],[5,4,0]]
输出:-1
解释:没有办法完成谜板

示例 3:

输入:board = [[4,1,2],[5,0,3]]
输出:5
解释:
最少完成谜板的最少移动次数是 5 ,
一种移动路径:
尚未移动: [[4,1,2],[5,0,3]]
移动 1 次: [[4,1,2],[0,5,3]]
移动 2 次: [[0,1,2],[4,5,3]]
移动 3 次: [[1,0,2],[4,5,3]]
移动 4 次: [[1,2,0],[4,5,3]]
移动 5 次: [[1,2,3],[4,5,0]]

题目详解:

class Solution {
private:
    // 定义每个位置与可以交换的相邻位置
    vector<vector<int>> neighbors = {{1, 3}, {0, 2, 4}, {1, 5}, {0, 4}, {1, 3, 5}, {2, 4}};
    // neighbors[x] 表示数字 x 所在位置与它可以交换的位置
public:
    int slidingPuzzle(vector<vector<int>>& board) {
        // 枚举状态,通过一次交换操作得到的状态
        auto get = [&](string& status) -> vector<string> {
            vector<string> ret;
            int x = status.find('0');  // 找到空缺的位置(数字 0)
            for (int y : neighbors[x]) {  // 遍历可以与 0 交换的位置
                swap(status[x], status[y]);  // 交换位置
                ret.push_back(status);  // 记录交换后的状态
                swap(status[x], status[y]);  // 再交换回来,保持原样
            }
            return ret;
        };
        
        // 将二维板转换为字符串形式,方便进行状态比较和转移
        string initial;
        for (int i = 0; i < 2; i++) {
            for (int j = 0; j < 3; j++) {
                initial += char(board[i][j] + '0');  // 将数字转为字符
            }
        }

        // 如果初始状态已经是目标状态 "123450",返回 0 步
        if (initial == "123450") {
            return 0;
        }

        // 使用 BFS 队列进行状态的广度优先搜索,保存当前状态和步数
        queue<pair<string, int>> q;
        q.emplace(initial, 0);
        
        // 哈希集合记录已经访问过的状态,避免重复搜索
        unordered_set<string> seen = {initial};

        // BFS 主循环
        while (!q.empty()) {
            auto [status, step] = q.front();
            q.pop();
            
            // 获取当前状态的所有可能后继状态
            for (auto&& next_status : get(status)) {
                if (!seen.count(next_status)) {  // 如果该状态没有被访问过
                    if (next_status == "123450") {  // 找到目标状态
                        return step + 1;
                    }
                    // 将新状态加入队列,步数 + 1
                    q.emplace(next_status, step + 1);
                    seen.insert(move(next_status));  // 将新状态标记为已访问
                }
            }
        }
        
        // 如果所有状态都搜索完毕还没有找到目标状态,返回 -1
        return -1;
    }
};

语法补充:

1.结构化绑定

(Structured Binding)它用于从 pair(或 tuple)等类型中直接解包多个值。这个语法是在 C++17 引入的。 

auto [status, step] = q.front();

相当于,可以简化代码:

pair<string, int> p = q.front();  // 取出队首元素
string status = p.first;  // 获取状态
int step = p.second;  // 获取步数

2.辅助函数:Lambda函数

具体可看文章:【C++基础】Lambda 函数 基础知识讲解学习及难点解析-CSDN博客

思路分析:

  1. 题目分析: 这个问题是一个经典的 状态转移问题,可以通过 广度优先搜索 (BFS) 来解决。我们需要在一个 2x3 的谜板中,进行数字的交换,使得最终的状态变成 [[1,2,3],[4,5,0]]。目标是找到解锁该谜板所需的最小步数。如果无法解锁,则返回 -1。

  2. 基本思路

    • 状态表示:使用一个字符串来表示谜板的状态,方便进行状态之间的转移。例如,初始状态 board = [[1,2,3],[4,5,0]] 可以表示为字符串 "123450"

    • 广度优先搜索:从初始状态出发,逐步扩展每一步的状态,直到找到目标状态 "123450",同时记录每个状态的步数。通过队列来保存待探索的状态,每次从队列中取出当前状态,生成所有可能的后继状态,并将未访问过的状态加入队列。

    • 状态转移:从当前状态出发,找到空白 0 所在的位置,然后交换 0 与它的相邻位置上的数字。每一次交换得到一个新的状态。

    • 优化:使用哈希集合 seen 来记录已经访问过的状态,避免重复计算。

  3. 具体步骤

    • 将二维谜板 board 转换为一个字符串 initial,方便后续的状态比较和操作。

    • 如果初始状态已经是目标状态 "123450",则返回 0,因为不需要任何操作。

    • 使用 BFS 从初始状态开始进行广度优先搜索,队列中存储当前状态及步数。

    • 对于每个状态,生成所有可能的后继状态,如果后继状态是目标状态,则返回当前步数 + 1。

    • 如果所有状态都搜索完毕而没有找到目标状态,则返回 -1。

  4. 时间和空间复杂度

    • 时间复杂度:每个状态的转移涉及到常数个操作,因此时间复杂度主要由状态数量决定。状态空间大小为 6!(即 720),因此时间复杂度为 O(720),即在最大状态数下,BFS 的复杂度是 O(N),其中 N 是状态的个数。

    • 空间复杂度:BFS 使用队列和哈希集合来存储状态,空间复杂度也是 O(N),即 O(720)。

相关文章:

  • Transformer架构
  • 【ARTS】2873.有序三元组中的最大值!
  • c语言修炼秘籍 - - 禁(进)忌(阶)秘(技)术(巧)【第五式】动态内存管理
  • JS实现AES和DES
  • 蓝桥杯刷题周计划(第四周)
  • 手撕LLM(二):从源码出发,探索LoRA加载、推理全流程
  • AudioFramework面试题
  • 【动态规划】线性dp
  • Windows事件日志清除
  • spring mvc异步请求 sse 大文件下载 断点续传下载Range
  • 【Kafka基础】topics命令行操作大全:进阶命令解析(2)
  • OpenSceneGraph (OSG) 开发入门
  • LeetCode 热题 100 堆
  • 快速通过简单代码了解装饰模式
  • 在Unity中,如果物体上的脚本丢失,可以通过编写一个自定义编辑器脚本来查找并删除这些丢失的组件
  • DeepSeek 教我 C++ (7) :常见的一些未定义UB
  • 基于论文的大模型应用:基于SmartETL的arXiv论文数据接入与预处理(一)
  • 嵌入式硬件篇---TOF陀螺仪SPI液晶屏
  • Xorg内存管理机制深度解析
  • P1025 [NOIP 2001 提高组] 数的划分(DFS)
  • 高端网站创建/企业宣传视频
  • 学做网站php吗/安卓系统优化app
  • 做高仿包的网站有哪些/短视频平台推广
  • 香港空间免费/宁波正规seo推广
  • 用织梦做网站费用/百度指数查询手机版
  • 做高档衣服的网站/上海优化价格