松下问童子,言师采药去。

只在此山中,云深不知处。——贾岛

题目:魔板 Magic Squares

网址:https://www.luogu.com.cn/problem/P2730

这是一张有8个大小相同的格子的魔板:

1 2 3 4

8 7 6 5

题目描述

我们知道魔板的每一个方格都有一种颜色。这8种颜色用前8个正整数来表示。可以用颜色的序列来表示一种魔板状态,规定从魔板的左上角开始,沿顺时针方向依次取出整数,构成一个颜色序列。

对于上图的魔板状态,我们用序列(1,2,3,4,5,6,7,8)来表示。

这是基本状态。

这里提供三种基本操作,分别用大写字母“A”,“B”,“C”来表示(可以通过这些操作改变魔板的状态):

“A”:交换上下两行;

“B”:将最右边的一列插入最左边;

“C”:魔板中央四格作顺时针旋转。

下面是对基本状态进行操作的示范:

A: 8 7 6 5

1 2 3 4

B: 4 1 2 3

5 8 7 6

C: 1 7 2 4

8 6 3 5

对于每种可能的状态,这三种基本操作都可以使用。

你要编程计算用最少的基本操作完成基本状态到目标状态的转换,输出基本操作序列。

输入格式

只有一行,包括8个整数,用空格分开(这些整数在范围 1——8 之间)不换行,表示目标状态。

输出格式

Line 1: 包括一个整数,表示最短操作序列的长度。

Line 2: 在字典序中最早出现的操作序列,用字符串表示,除最后一行外,每行输出60个字符。

输入样例
2 6 8 4 5 7 3 1
输出样例
7
BCABCCB
说明

选自USACO Training Section 3.2

做完八数码问题,不难理解这道题可以使用广搜解决(毕竟时间复杂度与八数码问题类似)。

像之前的八数码问题一样,我们把状态定义成一个八位数,代码中简单的使用STL中map和set进行记录与判重。

留意:本题有一个陷肼,如果使用从上到下、从左向右进行加和操作确定状态时,初状态应为:12348765!

代码如下:

#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<queue>
#include<map>
#include<set>
using namespace std;
const int rot[4] = {2, 1, 5, 6};
int st = 12348765, ed = 0, s[10];
map <int, int> d, f;
set <int> vis;
void decode(int state, int *p)
{
for(int i = 7; i >= 0; -- i)
{
p[i] = state % 10;
state /= 10;
}
//将十进制编码转化为数组
return;
}
int encode(int *p)
{
int cnt = 0;
for(int i = 0; i < 8; ++ i)
{
cnt = (cnt << 1) + (cnt << 3) + p[i];
}
//将数组转化为十进制8位编码
return cnt;
}
void flip(int cur, int *p)
{
switch(cur)
{
case 0 :
{
for(int i = 0; i < 4; ++ i)
{
swap(p[i], p[i + 4]);
}
break;
}
case 1 :
{
for(int i = 2; i > -1; -- i)
{
swap(p[i], p[i + 1]);
swap(p[i + 4], p[i + 5]);
}
break;
}
case 2 :
{
for(int i = 0; i < 3; ++ i)
{
swap(p[rot[i]], p[rot[i + 1]]);
}
break;
}
}
return;
}
void print_ans(int state)
{
if(state == st) return;
int prev_state = 0;
decode(state, s);
switch(f[state])
{
case 0:
{
for(int i = 0; i < 4; ++ i)
{
swap(s[i], s[i + 4]);
}
break;
}
case 1:
{
for(int i = 0; i < 3; ++ i)
{
swap(s[i], s[i + 1]);
swap(s[i + 4], s[i + 5]);
}
break;
}
case 2:
{
for(int i = 2; i >= 0; -- i)
{
swap(s[rot[i + 1]], s[rot[i]]);
}
break;
}
}
for(int i = 0; i < 8; ++ i)
{
prev_state = prev_state * 10 + s[i];
}
memset(s, 0, sizeof(s));
print_ans(prev_state);
printf("%c", f[state] + 'A');
return;
}
void bfs()
{
//特判 初始状态 和 目标状态 相同
if(st == ed) {
puts("0");
return;
} f.clear(), vis.clear();
queue <int> Q;
while(!Q.empty()) Q.pop(); Q.push(st);
vis.insert(st);
int copy[8] = {}, now, next; while(!Q.empty())
{
now = Q.front();
Q.pop();
decode(now, s);
memcpy(copy, s, sizeof(copy));
for(int i = 0; i < 3; ++ i)
{
flip(i, s);
next = encode(s);
memcpy(s, copy, sizeof(s)); if(vis.count(next)) continue;
d[next] = d[now] + 1;
f[next] = i;
if(next == ed)
{
printf("%d\n", d[next]);
print_ans(next);
return;
}
Q.push(next);
vis.insert(next);
}
}
return;
}
int main()
{
memset(s, 0, sizeof(s));
for(int i = 0; i < 4; ++ i) scanf("%d", &s[i]);
//目标状态处理:
for(int i = 7; i >= 4; -- i) scanf("%d", &s[i]);
//留意陷阱呢
for(int i = 0; i < 8; ++ i)
{
ed = (ed << 1) + (ed << 3) + s[i];
}
bfs();
return 0;
}

[USACO3.2]魔板 Magic Squares的更多相关文章

  1. 「一本通 1.4 例 2」[USACO3.2]魔板 Magic Squares

    [USACO3.2]魔板 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. 哈希+Bfs【P2730】 魔板 Magic Squares

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

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

    LG 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 题目描述 ...

  6. [洛谷P2730] 魔板 Magic Squares

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

  7. 魔板 Magic Squares(广搜,状态转化)

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

  8. P2730 魔板 Magic Squares

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

  9. 洛谷P2730 [IOI]魔板 Magic Squares

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

随机推荐

  1. 轻轻松松了解Java的初始化方法(含对象数组的小问题)

    Java基础复习6-初始化 之前讲过了类的一些基础,想必大家都知道了类的定义 什么?你忘了? 下面给你一个例子快速复习 class AClass{ int a; boolean b; void get ...

  2. 1063 Set Similarity (25分)

    Given two sets of integers, the similarity of the sets is defined to be /, where N​c​​ is the number ...

  3. Html 慕课园编程练习10-1

    23:10:25 2019-08-14 自己写的这个好丑.... 题目:利用之前我们学过的JavaScript知识,实现选项卡切换的效果. 效果图: (另外 这个动图是怎么插入的 用url就行 复制就 ...

  4. Spring MVC 笔记--配置基于JavaConfig

    主要使用基于 JavaConfig 方式配置 配置 DispatcherServlet 通过继承抽象类AbstractAnnotationConfigDispatcherServletInitiali ...

  5. Visual Studio Code 1.44 解决中文代码显示乱码问题(小白图文教程)

    现今主流的计算机中文字符编码方案是:GBK和UTF-8. 不同编码方案使用不同的字符集,GBK字符集在中文字符长度和字符数量上存在绝对优势,但对国外字符并不支持.所以,完全面向国内的程序/网页使用的是 ...

  6. 使用appium框架测试安卓app时,获取toast弹框文字时,前一步千万不要加time.sleep等等待时间。

    使用appium框架测试安卓app时,如果需要获取toast弹框的文案内容,那么再点击弹框按钮之前,一定记得千万不要加time.sleep()等待时间,否则有延迟,一直获取不到: 获取弹框的代码: m ...

  7. centos 64位安装jpeg-6b

    先安装libtool和libtool-ltdl-devel 用find命令查找config.sub和config.guess文件 find / -name config.sub find / -nam ...

  8. java第十九天,Collections集合工具类的使用

    Collections Java中集合都实现了Collection接口,那么针对集合一些特定的功能,有没有一个接口或类能够统一的集成一些集合必要的功能呢?当然能.它就是--Collections集合工 ...

  9. Pytest系列(14)- 配置文件pytest.ini的详细使用

    如果你还想从头学起Pytest,可以看看这个系列的文章哦! https://www.cnblogs.com/poloyy/category/1690628.html 前言 pytest配置文件可以改变 ...

  10. python连接mysql中文数据编码

    系统是win7 x64 Python 2.7.6的site.py里面编码设定为 utf-8 py文件首行指定 #coding:utf-8 MySQL 5.5.38安装时指定代码为utf-8 peewe ...