题解【洛谷P2730】魔板 Magic Squares
首先我们可以发现,在每一次 BFS 时按照 \(A→B→C\) 的顺序枚举遍历肯定是字典序最小的。
然后就是普通的 BFS 了。
我们考虑使用 \(\text{STL map}\) 来存储起点状态到当前状态所需的最少步数,以及到达它的上一个状态与上一个操作代号。
具体实现可参考代码。
#include <bits/stdc++.h>
using namespace std;
int n, m;
string start = "12345678", endd; //起始状态和目标状态
map <string, int> dist; //存储到达当前状态的最少步数
map <string, pair <char, string> > ans; //到达当前状态的上一个状态和操作代号
string q[100003], sum; //BFS 时的队列 和 答案
int hh, tt; //队头和队尾指针
char g[3][5]; //临时矩阵,转移的时候用
inline void getjuzhen(string x) //从压缩的状态变成矩阵
{
for (int i = 1; i <= 4; i+=1) g[1][i] = x[i - 1]; //第一排
g[2][4] = x[4], g[2][3] = x[5], g[2][2] = x[6], g[2][1] = x[7]; //第二排
}
inline string getzhuangtai() //从矩阵压缩成状态
{
string now = "";
for (int i = 1; i <= 4; i+=1) now = now + g[1][i]; //第一排
for (int j = 4; j >= 1; j-=1) now = now + g[2][j]; //第二排
return now;
}
inline string getA(string x) //A 操作
{
getjuzhen(x);
for (int i = 1; i <= 4; i+=1) swap(g[1][i], g[2][i]); //将第一行与第二行交换
return getzhuangtai();
}
inline string getB(string x) //B 操作
{
getjuzhen(x);
swap(g[1][1], g[1][4]), swap(g[2][1], g[2][4]);
swap(g[1][2], g[1][4]), swap(g[2][2], g[2][4]);
swap(g[1][3], g[1][4]), swap(g[2][3], g[2][4]);
//依次交换每一列
return getzhuangtai();
}
inline string getC(string x) //C 操作
{
getjuzhen(x);
swap(g[1][2], g[1][3]);
swap(g[1][2], g[2][3]);
swap(g[1][2], g[2][2]);
//中间的 4 个数依次交换
return getzhuangtai();
}
inline void bfs(string s, string t) //BFS
{
if (s == t) return; //如果目标状态与开始状态相同就不要搜索了
hh = tt = 0;
q[0] = s; //队列中只有 1 个元素
while (hh <= tt) //队列不为空
{
string c = q[hh++]; //取出队头元素
string h[3];
h[0] = getA(c); //A 操作
h[1] = getB(c); //B 操作
h[2] = getC(c); //C 操作
for (int i = 0; i < 3; i+=1) //枚举每一个操作
{
if (dist[h[i]] == 0) //如果当前状态还没有遍历过
{
dist[h[i]] = dist[c] + 1; //记录最少步数
ans[h[i]] = (make_pair)(i + 'A', c); //记录转移过来的操作代号和状态
if (h[i] == t) return; //找到了目标状态
q[++tt] = h[i]; //加入队列
}
}
}
}
int main()
{
for (int i = 1; i <= 8; i+=1)
{
int u; cin >> u;
endd = endd + (char)(u + '0'); //目标状态
}
bfs(start, endd);
cout << dist[endd] << endl; //输出最少步数
if (dist[endd] == 0) return 0; //注意特判
while (endd != start) //推出每一步的操作
{
sum = sum + ans[endd].first; //记录每一步的操作
endd = ans[endd].second; //向前推
}
reverse(sum.begin(), sum.end()); //记得要反转,因为我们存储的操作是反序的
cout << sum << endl; //输出操作序列
return 0;
}
题解【洛谷P2730】魔板 Magic Squares的更多相关文章
- 洛谷 P2730 魔板 Magic Squares 解题报告
P2730 魔板 Magic Squares 题目背景 在成功地发明了魔方之后,鲁比克先生发明了它的二维版本,称作魔板.这是一张有8个大小相同的格子的魔板: 1 2 3 4 8 7 6 5 题目描述 ...
- 洛谷 P2730 魔板 Magic Squares
P2730 魔板 Magic Squares 题目背景 在成功地发明了魔方之后,鲁比克先生发明了它的二维版本,称作魔板.这是一张有8个大小相同的格子的魔板: 1 2 3 4 8 7 6 5 题目描述 ...
- [洛谷P2730] 魔板 Magic Squares
洛谷题目链接:魔板 题目背景 在成功地发明了魔方之后,鲁比克先生发明了它的二维版本,称作魔板.这是一张有8个大小相同的格子的魔板: 1 2 3 4 8 7 6 5 题目描述 我们知道魔板的每一个方格都 ...
- 洛谷 - P2730 - 魔板 Magic Squares - bfs
写状态转移弄了很久,老了,不记得自己的数组是怎么标号的了. #include <bits/stdc++.h> using namespace std; #define ll long lo ...
- 洛谷P2730 魔板 [广搜,字符串,STL]
题目传送门 魔板 题目背景 在成功地发明了魔方之后,鲁比克先生发明了它的二维版本,称作魔板.这是一张有8个大小相同的格子的魔板: 1 2 3 4 8 7 6 5 题目描述 我们知道魔板的每一个方格都有 ...
- P2730 魔板 Magic Squares
题目背景 在成功地发明了魔方之后,鲁比克先生发明了它的二维版本,称作魔板.这是一张有8个大小相同的格子的魔板: 1 2 3 4 8 7 6 5 题目描述 我们知道魔板的每一个方格都有一种颜色.这8种颜 ...
- P2730 魔板 Magic Squares (搜索)
题目链接 Solution 这道题,我是用 \(map\) 做的. 具体实现,我们用一个 \(string\) 类型表示任意一种情况. 可以知道,排列最多只有 \(8!\) 个. 然后就是直接的广搜了 ...
- 哈希+Bfs【P2730】 魔板 Magic Squares
没看过题的童鞋请去看一下题-->P2730 魔板 Magic Squares 不了解康托展开的请来这里-->我这里 至于这题为什么可以用康托展开?(瞎说时间到. 因为只有8个数字,且只有1 ...
- 【简●解】 LG P2730 【魔板 Magic Squares】
LG P2730 [魔板 Magic Squares] [题目背景] 在成功地发明了魔方之后,鲁比克先生发明了它的二维版本,称作魔板.这是一张有8个大小相同的格子的魔板: 1 2 3 4 8 7 6 ...
- [USACO3.2]魔板 Magic Squares
松下问童子,言师采药去. 只在此山中,云深不知处.--贾岛 题目:魔板 Magic Squares 网址:https://www.luogu.com.cn/problem/P2730 这是一张有8个大 ...
随机推荐
- nginx的进程结构
nginx分为单进程和多进程,默认是多进程 进程架构: 父进程master process 子进程worker process和cache manager cache loader 高可用性 高可靠 ...
- 20194653 面向对象基础3——static、this、包总结
题目1:编写一个类Computer,类中含有一个求n的阶乘的方法.将该类打包,并在另一包中的Java文件App.java中引入包,在主类中定义Computer类的对象,调用求n的阶乘的方法(n值由参数 ...
- winform应用如何发布(不用打包)、并提醒用户自动更新
环境:VS2019 community C# winform 应用程序 设计应用程序界面 编写对应代码 使用PS设计程序标识ICON F4打开属性: 设置ICON 设置背景 打开项目属性 打开“发 ...
- Go语言之路—博客目录
Go语言介绍 为什么你应该学习Go语言? 开发环境准备 从零开始搭建Go语言开发环境 VS Code配置Go语言开发环境 Go语言基础 Go语言基础之变量和常量 Go语言基础之基本数据类型 Go语言基 ...
- GBM,XGBoost,LightGBM
GBM如何调参:https://www.analyticsvidhya.com/blog/2016/02/complete-guide-parameter-tuning-gradient-boosti ...
- Codeforces_456_A
http://codeforces.com/problemset/problem/456/A 按价格排序,比较质量. #include<cstdio> #include<algori ...
- 11--Java--JDBC知识梳理
JDBC 一.概述:JDBC(java database connection),使用java语言连接数据库,是java提供一套操作数据库的接口(标准),实现对数据库的统一访问,是一个java引用应用 ...
- Linux学习2-云服务器上安装java和tomcat环境
在linux上部署java的项目,首先要安装JDK和Tomcat,具体要求怎么操作呢,我们一起来学习吧! JDK的安装步骤如下: 1.首先我们从官网下载jdk-8u231-linux-x64.rpm安 ...
- Java TreeSet的使用
1.TreeSe自带排序的set,没有重复元素. 2.TreeSet 如果构造函数中没有使用比较器,那在装载的对象类中要实现Comparable 接口. 3.TreeSet 使用初始化比较器的方式. ...
- 终于成功部署 Kubernetes HPA 基于 QPS 进行自动伸缩
昨天晚上通过压测验证了 HPA 部署成功了. 所使用的 HPA 配置文件如下: apiVersion: autoscaling/v2beta2 kind: HorizontalPodAutoscale ...