题面

首先我们可以发现,在每一次 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的更多相关文章

  1. 洛谷 P2730 魔板 Magic Squares 解题报告

    P2730 魔板 Magic Squares 题目背景 在成功地发明了魔方之后,鲁比克先生发明了它的二维版本,称作魔板.这是一张有8个大小相同的格子的魔板: 1 2 3 4 8 7 6 5 题目描述 ...

  2. 洛谷 P2730 魔板 Magic Squares

    P2730 魔板 Magic Squares 题目背景 在成功地发明了魔方之后,鲁比克先生发明了它的二维版本,称作魔板.这是一张有8个大小相同的格子的魔板: 1 2 3 4 8 7 6 5 题目描述 ...

  3. [洛谷P2730] 魔板 Magic Squares

    洛谷题目链接:魔板 题目背景 在成功地发明了魔方之后,鲁比克先生发明了它的二维版本,称作魔板.这是一张有8个大小相同的格子的魔板: 1 2 3 4 8 7 6 5 题目描述 我们知道魔板的每一个方格都 ...

  4. 洛谷 - P2730 - 魔板 Magic Squares - bfs

    写状态转移弄了很久,老了,不记得自己的数组是怎么标号的了. #include <bits/stdc++.h> using namespace std; #define ll long lo ...

  5. 洛谷P2730 魔板 [广搜,字符串,STL]

    题目传送门 魔板 题目背景 在成功地发明了魔方之后,鲁比克先生发明了它的二维版本,称作魔板.这是一张有8个大小相同的格子的魔板: 1 2 3 4 8 7 6 5 题目描述 我们知道魔板的每一个方格都有 ...

  6. P2730 魔板 Magic Squares

    题目背景 在成功地发明了魔方之后,鲁比克先生发明了它的二维版本,称作魔板.这是一张有8个大小相同的格子的魔板: 1 2 3 4 8 7 6 5 题目描述 我们知道魔板的每一个方格都有一种颜色.这8种颜 ...

  7. P2730 魔板 Magic Squares (搜索)

    题目链接 Solution 这道题,我是用 \(map\) 做的. 具体实现,我们用一个 \(string\) 类型表示任意一种情况. 可以知道,排列最多只有 \(8!\) 个. 然后就是直接的广搜了 ...

  8. 哈希+Bfs【P2730】 魔板 Magic Squares

    没看过题的童鞋请去看一下题-->P2730 魔板 Magic Squares 不了解康托展开的请来这里-->我这里 至于这题为什么可以用康托展开?(瞎说时间到. 因为只有8个数字,且只有1 ...

  9. 【简●解】 LG P2730 【魔板 Magic Squares】

    LG P2730 [魔板 Magic Squares] [题目背景] 在成功地发明了魔方之后,鲁比克先生发明了它的二维版本,称作魔板.这是一张有8个大小相同的格子的魔板: 1 2 3 4 8 7 6 ...

  10. [USACO3.2]魔板 Magic Squares

    松下问童子,言师采药去. 只在此山中,云深不知处.--贾岛 题目:魔板 Magic Squares 网址:https://www.luogu.com.cn/problem/P2730 这是一张有8个大 ...

随机推荐

  1. nginx的四个主要组成部分

    1.nginx二进制可执行文件 · 由各模块源码编译出的一个文件 2.nginx.conf配置文件 · 控制nginx的行为 3.access.log访问日志 . 记录每一条http请求信息 4.er ...

  2. 分析Ajax爬取今日头条街拍美图-崔庆才思路

    站点分析 源码及遇到的问题 代码结构 方法定义 需要的常量 关于在代码中遇到的问题 01. 数据库连接 02.今日头条的反爬虫机制 03. json解码遇到的问题 04. 关于response.tex ...

  3. 网络最大流(EK)

    以前在oi中见到网络流的题都是直接跳过,由于本蒟蒻的理解能力太弱,导致网络流的学习不断推迟甚至被安排在了tarjan之后,原本计划于学习完最短路后就来学网络流的想法也随之破灭,在看完众多大佬 的博客后 ...

  4. hdu 1159求最长公共子序列

    题目描述:给出两个字符串,求两个字符串的公共子序列(不是公共子串,不要求连续,但要符合在原字符串中的顺序) in: abcfbc abfcab programming contest abcd mnp ...

  5. Codeforces_492_E

    http://codeforces.com/problemset/problem/492/E 题目规定了gcd=1,可以在纸上模拟一下,发现每一个起点,都会经历过n个点,n个点都是不同行不同列.可以把 ...

  6. Codeforces 922 E Birds (背包dp)被define坑了的一题

    网页链接:点击打开链接 Apart from plush toys, Imp is a huge fan of little yellow birds! To summon birds, Imp ne ...

  7. 题解【RQNOJ PID497 0/1字串问题】

    \[ \texttt{Description} \] 编程找出符合下列条件的字符串:①字符串中仅包含 0 和 1 两个字符:②字符串的长度为 n :③字符串中不包含连续重复三次的子串. \[ \tex ...

  8. 题解【CF1303D Fill The Bag】

    \[ \texttt{Preface} \] 不开 long long 见祖宗. \[ \texttt{Description} \] 你有一个 \(n\) 码的袋子,你还有 \(m\) 个盒子,第 ...

  9. MySQL复制(四)—多源(主)复制

    (一)多主复制概述 MySQL从5.7版本开启支持多主复制,所谓多主复制,是将多个主库的数据复制到一个从库中.通常用于数据仓库整合数据,比如OLTP系统为了分散业务压力,对数据库进行分库分表,当要对数 ...

  10. Update、Insert注入技巧

    title: Update.Insert注入技巧 date: 2017-10-23 18:07:57 tags: ["注入"] 审计了不少代码,再看代码的时候最多出现的就是注入,很 ...