【BFS最小步数】魔板题解
魔板题解
题目传送门
题目传送门
一、题目描述
Rubik先生发明了魔板的二维版本,这是一个有8个格子的板子,初始状态为:
1 2 3 4
8 7 6 5
我们可以用三种操作来改变魔板状态:
- A:交换上下两行
- B:将最右边一列插入到最左边
- C:魔板中央的4个数顺时针旋转
给定一个目标状态,要求找出从初始状态到目标状态的最短操作序列,如果有多个解,输出字典序最小的那个。
二、题目分析
这是一个典型的状态空间搜索问题,我们需要找到从初始状态到目标状态的最短路径。由于状态数量有限(8! = 40320种可能状态),可以使用广度优先搜索(BFS)来求解。
三、解题思路
- 使用BFS从初始状态开始搜索
- 对于每个状态,尝试三种操作生成新状态
- 记录每个状态的来源和操作,以便最后回溯路径
- 当找到目标状态时,根据记录的信息回溯操作序列
- 由于BFS按层扩展,第一个找到的解就是最短的
- 按照A、B、C的顺序尝试操作,保证字典序最小
四、算法讲解
广度优先搜索(BFS)
- 算法原理:BFS是一种图搜索算法,它从起始节点开始,逐层向外扩展,直到找到目标节点。在状态空间搜索中,每个节点代表一个状态,边代表操作。
- 使用场景:适用于寻找无权图中的最短路径问题,如本题中的最少操作次数。
- 实现步骤:
- 初始化队列,加入起始状态
- 记录每个状态的访问情况和距离
- 对于队列中的每个状态,尝试所有可能的操作
- 将未访问过的新状态加入队列
- 直到找到目标状态为止
结合例子
以样例输入2 6 8 4 5 7 3 1为例:
- 初始状态:1 2 3 4 5 6 7 8
- 通过BFS逐步扩展,最终找到目标状态
- 回溯操作序列得到
BCABCCB
五、代码实现
#include <bits/stdc++.h>
using namespace std;
string Start, End;
char g[2][4];
unordered_map<string, pair<char, string>> pre; // 记录前驱状态和操作
unordered_map<string, int> dist; // 记录距离// 将字符串表示的状态设置到二维数组中
void setString(string t)
{for (int i = 0; i < 4; i++)g[0][i] = t[i];for (int i = 7, j = 0; j < 4; i--, j++)g[1][j] = t[i];
}// 从二维数组获取字符串表示的状态
string getString()
{string res;for (int i = 0; i < 4; i++)res += g[0][i];for (int i = 3; i >= 0; i--)res += g[1][i];return res;
}// 操作A:交换上下两行
string move0(string t)
{setString(t);for (int i = 0; i < 4; i++)swap(g[0][i], g[1][i]);return getString();
}// 操作B:将最右边的一列插入到最左边
string move1(string t)
{setString(t);char v0 = g[0][3], v1 = g[1][3];for (int i = 3; i > 0; i--)g[0][i] = g[0][i - 1], g[1][i] = g[1][i - 1];g[0][0] = v0, g[1][0] = v1;return getString();
}// 操作C:魔板中央的4个数顺时针旋转
string move2(string t)
{setString(t);char v = g[0][1];g[0][1] = g[1][1];g[1][1] = g[1][2];g[1][2] = g[0][2];g[0][2] = v;return getString();
}// BFS搜索最短路径
int bfs(string Start, string end)
{if (Start == end)return 0;queue<string> q;q.push(Start);dist[Start] = 0;while (q.size()){auto t = q.front();q.pop();string m[3];m[0] = move0(t); // 尝试操作Am[1] = move1(t); // 尝试操作Bm[2] = move2(t); // 尝试操作Cfor (int i = 0; i < 3; i++)if (!dist.count(m[i])){dist[m[i]] = dist[t] + 1;pre[m[i]] = {'A' + i, t}; // 记录前驱和操作q.push(m[i]);if (m[i] == end)return dist[m[i]];}}return -1;
}void solve()
{for (int i = 0; i < 8; i++){int x;cin >> x;End += char(x + '0');}Start = "12345678"; // 初始状态int step = bfs(Start, End);cout << step << "\n";string res;while (End != Start){res += pre[End].first; // 获取操作End = pre[End].second; // 回溯到前驱状态}reverse(res.begin(), res.end()); // 反转得到正确顺序if (step > 0)cout << res << "\n";
}int main()
{ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);solve();return 0;
}
六、重点细节
- 状态表示:使用字符串表示魔板状态,前四个字符是第一行,后四个字符是第二行的逆序。
- 操作实现:三种操作通过二维数组转换实现,注意操作C的旋转方向。
- 字典序处理:按照A、B、C的顺序尝试操作,保证找到的第一个解就是字典序最小的。
- 路径记录:使用
pre哈希表记录每个状态的前驱和操作,便于回溯路径。 - 边界处理:当初始状态就是目标状态时直接返回0。
七、复杂度分析
- 时间复杂度:O(8!) = O(40320),因为最多有8!种不同的状态。
- 空间复杂度:O(8!),需要存储所有可能状态的访问信息和前驱关系。
八、总结
本题通过BFS搜索状态空间,利用哈希表记录状态信息和路径,是一种典型的图搜索问题。关键在于:
- 合理表示状态
- 正确实现三种操作
- 高效记录和回溯路径
- 处理字典序要求
这种BFS方法适用于类似的"最少操作次数"问题,如八数码问题等。
相关文章:
【BFS最小步数】魔板题解
魔板题解 题目传送门 题目传送门 一、题目描述 Rubik先生发明了魔板的二维版本,这是一个有8个格子的板子,初始状态为: 1 2 3 4 8 7 6 5我们可以用三种操作来改变魔板状态: A:交换上下两行B:将最右边一…...
搭建K8S-1.23
0、简介 这里只用3台服务器来做一个简单的集群 地址主机名192.168.160.40kuber-master-1192.168.160.41kuber-master-2192.168.160.42kuber-node-1 1、关闭三个服务 (1)防火墙 systemctl stop firewalld (2)Selinux setenf…...
AOP在SpringBoot项目中的简单使用场景
SpringBoot AOP的简单使用 添加DTO添加controller(同包不同类)控制器1控制器2 AOP场景演示1. 对某package下的所有接口进行方法执行前逻辑校验新增切面,编写处理逻辑 2. 对某controller类下的所有接口进行方法执行前逻辑校验新增切面,编写处理逻辑 3. 对…...
windows如何安装wkhtmltoimage 给PHP使用根据HTML生成图片
windows如何安装wkhtmltoimage 给PHP使用 在Windows系统上安装wkhtmltoimage以便在PHP中使用,通常涉及到以下几个步骤: 下载wkhtmltoimage 首先,你需要从wkhtmltopdf的官方网站( https://wkhtmltopdf.org/downloads.html …...
代码仓库使用git lfs上传模型文件
一 Git LFS是什么 它主要是用来处理大文件的,比如模型文件通常都很大,超过100MB的话,用普通的Git上传可能会出问题,所以必须用LFS。 二 具体步骤 Windows环境下使用Git LFS上传大模型文件到代码仓库: 2.1ÿ…...
AI比人脑更强,因为被植入思维模型【42】思维投影思维模型
giszz的理解:本质和外在。我们的行为举止,都是我们的内心的表现。从外边可以看内心,从内心可以判断外在。曾国藩有7个识人的方法,大部分的人在他的面前如同没穿衣服一样。对于我们自身的启迪,我认为有四点&…...
Git 从入门到精通(开源协作特别版)
🧠 Git 从入门到精通(开源协作特别版) ✅ 基础命令 🧰 高级用法 🛠️ 开源实战技巧 🌍 GitHub 社区协作 适合:从0开始 → 熟练开发者 → 参与/维护开源项目 🔰 第1章:…...
spring-cloud-alibaba-nacos-config使用说明
一、核心功能与定位 Spring Cloud Alibaba Nacos Config 是 Spring Cloud Alibaba 生态中的核心组件之一,专为微服务架构提供动态配置管理能力。它通过整合 Nacos 的配置中心功能,替代传统的 Spring Cloud Config,提供更高效的配置集中化管理…...
C# Winform 入门(9)之如何封装并调用dll
封装dll 首先创建 .Net平台 类库 using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks;namespace _09.Encapsulation_dll {public class Program{/// <summary>/// 求两个double类型的数值的和/// &l…...
vue3中ref、reactive的使用示例
ref 1、导入 import { ref } from "vue"; 2、定义 // 报告表格数据 const reportTableData ref<Report[]>([]); 3、赋值 // 获取报告信息 let result await reportDataByOuterApplyIdService(tableSelectedRow.value?.outerApplyId); reportTable…...
【嵌入式系统设计师】知识点:第2章 嵌入式系统硬件基础知识
提示:“软考通关秘籍” 专栏围绕软考展开,全面涵盖了如嵌入式系统设计师、数据库系统工程师、信息系统管理工程师等多个软考方向的知识点。从计算机体系结构、存储系统等基础知识,到程序语言概述、算法、数据库技术(包括关系数据库、非关系型数据库、SQL 语言、数据仓库等)…...
Vue2_Vue.js教程
目录 一、Vue.js安装 1、独立版本 2、CDN 方法 3、npm 方法 二、Vue Al编程助手 三、Vue.js目录结构 目录解析 四、Vue.js 起步 1.如何定义数据对象和方法并渲染进页面 五、Vue.js 模板语法 插值 文本_{{}} Html_v-html 指令 属性_v-bind (数据传输工具)指令 表…...
【工业场景】用YOLOv12实现饮料类别识别
饮料类别识别任务的意义在于帮助人们更快速地识别和区分不同类型的饮料,从而提高消费者的购物体验和满意度。对于商家而言,饮料类别识别可以帮助他们更好地管理库存、优化货架布局和预测销售趋势,从而提高运营效率和利润。此外,饮…...
从小米汽车事故反思 LabVIEW 开发
近期,小米汽车的一起严重事故引发了社会各界的广泛关注。这起事故不仅让我们对智能汽车的安全性产生了深深的思考,也为 LabVIEW 开发领域带来了诸多值得汲取的知识与领悟。 在智能汽车领域,尤其是涉及到智能驾驶辅助系统时,安全是…...
oracle WAIT 和 NOWAIT
在 Oracle 数据库中,WAIT 和 NOWAIT 是与 锁(Lock) 相关的关键选项,用于控制事务或操作在请求资源时的等待行为。以下是它们的详细说明和应用场景。 1. NOWAIT 选项 作用: 当请求资源(如表、行)…...
Vue3+Vite+TypeScript+Element Plus开发-04.静态菜单设计
系列文档目录 Vue3ViteTypeScript安装 Element Plus安装与配置 主页设计与router配置 静态菜单设计 Pinia引入 文章目录 目录 系列文档目录 文章目录 前言 一、Aside设计 二、动态增加菜单 三.布局引用在Main中显示 参考文献: 前言 在本系列文档中&…...
从代码学习深度学习 - LSTM PyTorch版
文章目录 前言一、数据加载与预处理1.1 代码实现1.2 功能解析二、LSTM介绍2.1 LSTM原理2.2 模型定义代码解析三、训练与预测3.1 训练逻辑代码解析3.2 可视化工具功能解析功能结果总结前言 深度学习中的循环神经网络(RNN)及其变种长短期记忆网络(LSTM)在处理序列数据(如文…...
大数据技术发展与应用趋势分析
大数据技术发展与应用趋势分析 文章目录 大数据技术发展与应用趋势分析1. 大数据概述2 大数据技术架构2.1 数据采集层2.2 数据存储层2.3 数据处理层2.4 数据分析层 3 大数据发展趋势3.1 AI驱动的分析与自动化3.2 隐私保护分析技术3.3 混合云架构的普及3.4 数据网格架构3.5 量子…...
与Linux操作系统相关的引导和服务
目录 一.Linux操作系统引导过程 1.1引导过程总览 1.2系统初始化进程 1.2.1init进程 1.2.2sysmted 1.3systemd单元类型 二.排除启动类故障 2.1MBR扇区故障 2.1.1故障原因 2.1.2故障现象 2.1.3解决办法 2.1.4模拟修复MBR扇区故障 1)添加新的硬盘 2)进行…...
STM32单片机入门学习——第16节: [6-4] PWM驱动LED呼吸灯PWM驱动舵机PWM驱动直流电机
写这个文章是用来学习的,记录一下我的学习过程。希望我能一直坚持下去,我只是一个小白,只是想好好学习,我知道这会很难,但我还是想去做! 本文写于:2025.04.05 STM32开发板学习——第16节: [6-4] PWM驱动LED呼吸灯&PWM驱动舵机&PWM驱…...
基础框架系列分享:一个通用的Excel报表生成管理框架
由于我们系统经常要生成大量的Excel报表(Word,PDF报表也有,另行分享),最初始他们的方案是,设计一个表,和Excel完全对应,然后读表,把数据填进去,这显然是非常不…...
Ansible(4)—— Playbook
目录 一、Ansible Playbook : 1、Play : 2、Playbook: 二、Ansible Playbook 格式: 1、空格: 2、破折号( - ): 3、Play 格式: 三、查找用于任务的模块: 1、模块…...
自学-C语言-基础-数组、函数、指针、结构体和共同体、文件
这里写自定义目录标题 代码环境:?问题思考:一、数组二、函数三、指针四、结构体和共同体五、文件问题答案: 代码环境: Dev C ?问题思考: 把上门的字母与下面相同的字母相连,线不能…...
Bash 花括号扩展 {start..end} 进阶使用指南——字典生成
Bash 的花括号扩展(brace expansion){start..end} 是一个强大而灵活的语法特性,用于生成特定序列或组合。它在脚本编写、爆破字典生成、文件批量操作以及模式匹配中有着广泛的应用。本文将从基础用法到高级技巧,带你全面掌握这一功…...
AGI大模型(10):prompt逆向-巧借prompt
1 提示词逆向 明确逆向提示词⼯程概念 我们可以给ChatGPT提供⼀个简洁的提示词,让它能够更准确地理解我们所讨论的“逆向提示词⼯程”是什么意思,并通过这个思考过程,帮它将相关知识集中起来,进⽽构建⼀个专业的知识领域 提示词:请你举⼀个简单的例⼦,解释⼀下逆向pro…...
蓝桥云客--团队赛
2.团队赛【算法赛】 - 蓝桥云课 问题描述 蓝桥杯最近推出了一项团队赛模式,要求三人组队参赛,并规定其中一人必须担任队长。队长的资格很简单:其程序设计能力值必须严格大于其他两名队友程序设计能力值的总和。 小蓝、小桥和小杯正在考虑报名…...
C-S模式之实现一对一聊天
天天开心!!! 文章目录 一、如何实现一对一聊天?1. 服务器设计2. 客户端设计3. 服务端代码实现4. 客户端代码实现5. 实现说明6.实验结果 二、改进常见的服务器高并发方案1. 多线程/多进程模型2. I/O多路复用3. 异步I/O(…...
[Deep-ML]Transpose of a Matrix(矩阵的转置)
Transpose of a Matrix(矩阵的转置) 题目链接: Transpose of a Matrix(矩阵的转置)https://www.deep-ml.com/problems/2 题目描述: 难度: easy(简单)。 分类&#…...
Java的Selenium的特殊元素操作与定位之select下拉框
如果页面元素是一个下拉框,我们可以将此web元素封装为Select对象 Select selectnew Select(WebElement element); Select对象常用api select.getOptions();//获取所有选项select.selectBylndex(index);//根据索引选中对应的元素select.selectByValue(value);//选…...
前端精度计算:Decimal.js 基本用法与详解
一、Decimal.js 简介 decimal.js 是一个用于任意精度算术运算的 JavaScript 库,它可以完美解决浮点数计算中的精度丢失问题。 官方API文档:Decimal.js 特性: 任意精度计算:支持大数、小数的高精度运算。 链式调用:…...
