魔板 (bfs+康托展开)
# 10027. 「一本通 1.4 例 2」魔板
【题目描述】
Rubik 先生在发明了风靡全球魔方之后,又发明了它的二维版本——魔板。这是一张有 888 个大小相同的格子的魔板:
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
对于每种可能的状态,这三种基本操作都可以使用。
你要编程计算用最少的基本操作完成基本状态到特殊状态的转换,输出基本操作序列。
【算法】
显然使用bfs,新知识点:cantor展开。注意魔板的输入顺序是顺时针。
【代码】
#include <bits/stdc++.h>
using namespace std;
int d[50010],rec[50010][2];
const int fac[]={ 1,1,2,6,24,120,720,5040,40320 };
struct state{ int now[2][4]; }st,ed;
queue<state> q;
int cantor(state& x) {
int tmp[8],res=0;
for(int i=0;i<2;i++)
for(int j=0;j<4;j++)
tmp[i*4+j]=x.now[i][j];
for(int i=0;i<8;i++) {
int s=0;
for(int j=i+1;j<8;j++) if(tmp[j]<tmp[i]) s++;
res+=s*fac[7-i];
}
return res;
}
void print(int x) {
if(x==cantor(st)) return;
print(rec[x][1]);
printf("%c",char(rec[x][0]));
}
int main() {
for(int i=0;i<4;i++) st.now[0][i]=i+1,scanf("%d",&ed.now[0][i]);
for(int i=0;i<4;i++) st.now[1][3-i]=4+i+1,scanf("%d",&ed.now[1][3-i]);
for(int i=0;i<=fac[8];i++) d[i]=-1;
d[cantor(st)]=0; q.push(st);
while(q.size()) {
state now=q.front(),tmp; q.pop();
if(cantor(now)==cantor(ed)) break;
//A
for(int i=0;i<2;i++) {
for(int j=0;j<4;j++)
tmp.now[(i+1)%2][j]=now.now[i][j];
}
if(d[cantor(tmp)]==-1) {
d[cantor(tmp)]=d[cantor(now)]+1;
rec[cantor(tmp)][0]='A';
rec[cantor(tmp)][1]=cantor(now);
q.push(tmp);
}
//B
tmp.now[0][0]=now.now[0][3],tmp.now[1][0]=now.now[1][3];
for(int i=0;i<2;i++) {
for(int j=0;j<3;j++)
tmp.now[i][j+1]=now.now[i][j];
}
if(d[cantor(tmp)]==-1) {
d[cantor(tmp)]=d[cantor(now)]+1;
rec[cantor(tmp)][0]='B';
rec[cantor(tmp)][1]=cantor(now);
q.push(tmp);
}
//C
memcpy(tmp.now,now.now,sizeof(now.now));
tmp.now[0][1]=now.now[1][1]; tmp.now[0][2]=now.now[0][1];
tmp.now[1][1]=now.now[1][2]; tmp.now[1][2]=now.now[0][2];
if(d[cantor(tmp)]==-1) {
d[cantor(tmp)]=d[cantor(now)]+1;
rec[cantor(tmp)][0]='C';
rec[cantor(tmp)][1]=cantor(now);
q.push(tmp);
}
}
printf("%d\n",d[cantor(ed)]);
print(cantor(ed));
return 0;
}
魔板 (bfs+康托展开)的更多相关文章
- hdu1430魔板(BFS+康托展开)
做这题先看:http://blog.csdn.net/u010372095/article/details/9904497 Problem Description 在魔方风靡全球之后不久,Rubik先 ...
- hdu.1430.魔板(bfs + 康托展开)
魔板 Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submis ...
- HDU 1430 魔板(康托展开+BFS+预处理)
魔板 Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submis ...
- HDU_1430——魔板,预处理,康托展开,置换,string类的+操作
Problem Description 在魔方风靡全球之后不久,Rubik先生发明了它的简化版——魔板.魔板由8个同样大小的方块组成,每个方块颜色均不相同,可用数字1-8分别表示.任一时刻魔板的状态可 ...
- HDU_1043 Eight 【逆向BFS + 康托展开 】【A* + 康托展开 】
一.题目 http://acm.hdu.edu.cn/showproblem.php?pid=1043 二.两种方法 该题很明显,是一个八数码的问题,就是9宫格,里面有一个空格,外加1~8的数字,任意 ...
- HDU_1430 魔板 【BFS+康托展开+置换】
一.题面 POJ1430 二.分析 该题与之前做的八数码不同,它是一个2*4的棋盘,并且没有空的区域.这样考虑的情况是很少的,依然结合康托展开,这时康托展开最多也只乘7的阶乘,完全可以BFS先预处理一 ...
- HDU - 1430 魔板 【BFS + 康托展开 + 哈希】
题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=1430 思路 我刚开始 想到的 就是 康托展开 但是这个题目是 多组输入 即使用 康托展开 也是会T的 ...
- HDU1430;魔板(BFS+康托展开)
传送门 题意 给出初始序列与终止序列,给出三种操作,问最少经过几次操作能使初始->终止,输出操作(字典序最小) 分析 字符串只有8个字符,使用康托展开. 1.BFS将所有序列从"123 ...
- hdu1430 魔板(康拓展开 bfs预处理)
魔板 Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Submiss ...
随机推荐
- 049:ORM常用Field详解(1)
常用字段: 在 Django 中,定义了一些 Field 来与数据库表中的字段类型来进行映射.以下将介绍那些常用的字段类型. AutoField: 映射到数据库中是 int 类型,可以有自动增长的特性 ...
- Ubuntu操作
1. 从windows 拷贝文件到Ubuntu: scp /c/Users/init/Downloads/pycharm-community-2019.3.tar.gz alice@192 ...
- appium环境666
一.安装JDK,配置JDK环境 appium下载地址: https://github.com/appium/appium-desktop/releases/ 百度搜索下载就行,这里分享一个下载链 ...
- linux运维、架构之路-Docker快速入门
一.Docker介绍 Docker是Docker.lnc公司开源的一个基于LXC技术之上构建的Container容器引擎,源代码托管在Github上,基于Go语言并遵从Apache2.0 ...
- css雪碧图-css精灵图
先将图片拼接在一张图上.类似实现的效果图 图片地址为合并后的图片地址,通过background-position调整背景图的位置.效果如: HTML: <div class="logo ...
- docker for windows 中挂载文件到容器
docker for windows版本: 宿主机:windows10 场景: 容器是基于microsoft/donet的webapi 想把宿主机的文件挂载到容器中,比方说:a.txt 命令如下: d ...
- mysql INSERT语句 语法
mysql INSERT语句 语法 作用:用于向表格中插入新的行. 语法:INSERT INTO 表名称 VALUES (值1, 值2,....)或者INSERT INTO table_name (列 ...
- 转:HTML5 History API 详解
从Ajax翻页的问题说起 请想象你正在看一个视频下面的评论,在翻到十几页的时候,你发现一个写得稍长,但非常有趣的评论.正当你想要停下滚轮细看的时候,手残按到了F5.然后,页面刷新了,评论又回到了第一页 ...
- C++11 lambda表达式小结
目录 简介 结构 return type parameter list capture list 值捕获和引用捕获 变量修改 隐式和显式捕获 捕获列表小结: problems 1.为什么需要使用mut ...
- Java Interger类,两对整数明明完全一样,为何一个输出true,一个输出false
package text; public class MethodOverload { public static void main(String[] args) { Integer i1=100; ...