题解【洛谷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个大 ...
随机推荐
- 代理IP批量验证程序
#include <afxinet.h> #include <afxwin.h> #include <iostream> #include <fstream& ...
- C语言实现银行家算法
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <stdbool.h& ...
- Linux/UNIX编程:实现简单 tee 命令
思路很简单,从标准输入文件描述符读入数据,然后同时向标准输出和参数指定的文件写出数据:如果加了 -a 选项,则以追加的方式向文件写出数据.还没了解 getopt() 函数就没判断参数是否合法. #in ...
- 《 Java 编程思想》CH06 访问权限控制
访问控制(或隐藏具体实现)与"最初的实现并不恰当"有关. 重构即重写代码,以使得它更可读,更容易理解,并因此而更具维护性. 在面向对象设计中需要考虑一个基本问题:"如何把 ...
- Java查找数组重复元素,并打印重复元素、重复次数、重复元素位置
面试题查找重复元素并打印重复次数和重复位置,一顿懵逼,回来死磕写下来,打印指定重复次数和最大次数,其他在此基础上可以再更新 package sort; import org.testng.annota ...
- 【python-leetcode713-双指针】乘积小于k的子数组
问题描述: 给定一个正整数数组 nums. 找出该数组内乘积小于 k 的连续的子数组的个数. 示例 1: 输入: nums = [10,5,2,6], k = 100输出: 8解释: 8个乘积小于10 ...
- Go语言实现:【剑指offer】正则表达式匹配
该题目来源于牛客网<剑指offer>专题. 请实现一个函数用来匹配包括 . 和 * 的正则表达式.模式中的字符.表示任意一个字符,而 * 表示它前面的字符可以出现任意次(包含0次). 在本 ...
- 某oa系统的审计
title: 某oa系统的审计 date: 2018-03-07 17:18:16 tags: --- 信呼OA 闲着没事,java学累了来整理下以前审的一个觉得很有意思的cms,这个作者写的比较灵活 ...
- iptables 实例
开启ip段192.168.1.0/24端的80口 开启ip段211.123.16.123/24端ip段的80口 # iptables -I INPUT -p tcp --dport 80 -j DRO ...
- OSPF配置实验(一)
单区域OSPF 命令: R1(config)#router ospf 1 //启动OSPF进程 R1(config-router)#router-id 1.1.1.1 // ...