# 10027. 「一本通 1.4 例 2」魔板

【题目描述】

Rubik 先生在发明了风靡全球魔方之后,又发明了它的二维版本——魔板。这是一张有 888 个大小相同的格子的魔板:

1 2 3 4
8 7 6 5

我们知道魔板的每一个方格都有一种颜色。这 8 种颜色用前 8 个正整数来表示。可以用颜色的序列来表示一种魔板状态,规定从魔板的左上角开始,沿顺时针方向依次取出整数,构成一个颜色序列。对于上图的魔板状态,我们用序列 $1,2,3,4,5,6,7,8$ 来表示。这是基本状态。

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

  • 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+康托展开)的更多相关文章

  1. hdu1430魔板(BFS+康托展开)

    做这题先看:http://blog.csdn.net/u010372095/article/details/9904497 Problem Description 在魔方风靡全球之后不久,Rubik先 ...

  2. hdu.1430.魔板(bfs + 康托展开)

    魔板 Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submis ...

  3. HDU 1430 魔板(康托展开+BFS+预处理)

    魔板 Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submis ...

  4. HDU_1430——魔板,预处理,康托展开,置换,string类的+操作

    Problem Description 在魔方风靡全球之后不久,Rubik先生发明了它的简化版——魔板.魔板由8个同样大小的方块组成,每个方块颜色均不相同,可用数字1-8分别表示.任一时刻魔板的状态可 ...

  5. HDU_1043 Eight 【逆向BFS + 康托展开 】【A* + 康托展开 】

    一.题目 http://acm.hdu.edu.cn/showproblem.php?pid=1043 二.两种方法 该题很明显,是一个八数码的问题,就是9宫格,里面有一个空格,外加1~8的数字,任意 ...

  6. HDU_1430 魔板 【BFS+康托展开+置换】

    一.题面 POJ1430 二.分析 该题与之前做的八数码不同,它是一个2*4的棋盘,并且没有空的区域.这样考虑的情况是很少的,依然结合康托展开,这时康托展开最多也只乘7的阶乘,完全可以BFS先预处理一 ...

  7. HDU - 1430 魔板 【BFS + 康托展开 + 哈希】

    题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=1430 思路 我刚开始 想到的 就是 康托展开 但是这个题目是 多组输入 即使用 康托展开 也是会T的 ...

  8. HDU1430;魔板(BFS+康托展开)

    传送门 题意 给出初始序列与终止序列,给出三种操作,问最少经过几次操作能使初始->终止,输出操作(字典序最小) 分析 字符串只有8个字符,使用康托展开. 1.BFS将所有序列从"123 ...

  9. hdu1430 魔板(康拓展开 bfs预处理)

    魔板 Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submiss ...

随机推荐

  1. CF839E Mother of Dragons 最大团 Bron-Kerbosch算法

    题意简述 给你一个\(n\)个节点的无向图\(G=\{V,E\}\)的邻接矩阵\(g\)和每个点的点权为\(s_i\),且\(\sum_{i=1}^n s_i = K\),要你求出\(\mathrm{ ...

  2. div中放入一个img元素导致div高度会多出几个像素

    在写代码的时候经常遇到这样一个问题,如果div里嵌套一个img元素且div的高度是由img的高度来撑开,那么div的高度总会比img的高度多3px.好了,废话不多说,直接给大家上代码. html代码: ...

  3. CSS3——制作正在加载页面loading...

    今天做了好多小东西,还挺开心的~ <!DOCTYPE html> <html lang="en"> <head> <meta charse ...

  4. CF1244C

    题目描述 给出n,p,w,d,求(x,y,z)使得 xw+yd=p x+y+z=n 其中d<w<10^5^ 题解 显然扩欧啊( 来自天国的long long y如果大于等于w,则显然可以把 ...

  5. 判断Java对象存活的算法、垃圾回收算法

    判断Java对象存活的算法 一.引用计数算法 给对象添加一个引用计数器,每当一个地方引用它的时候就将计数器加1,当引用失效的时候就将计数器减1,任何时刻计数器为0的对象都不可再被使用.这种算法虽然简单 ...

  6. mysql JOIN关键字 语法

    mysql JOIN关键字 语法 作用:用于根据两个或多个表中的列之间的关系,从这些表中查询数据.大理石量具 说明:数据库中的表可通过键将彼此联系起来.主键(Primary Key)是一个列,在这个列 ...

  7. sh_03_注释

    sh_03_注释 # 这是第一个注释 print("hello hello") """ 这是一个多行注释 .... .... .... 注释结束了 & ...

  8. 4.16中Montage的一些变化

    用4.16版本跟着网上的几篇Montage教程(分别是对应4.6和4.8版本)做,遇到各种各样的问题,经过各种搜索和分析,最终终于搞定了. 4.16版的Montage和之前有了不小的变化,总结如下: ...

  9. 500 cannot be cast to javax.xml.registry.infomodel

    在使用mybatis的时候每次一调用一个返回User类型的sql时,总是会报错如下: com.xx.all.domain.User cannot be cast to javax.xml.regist ...

  10. Qt 之 ZIP开源库 QuaZIP

    2013-10-31 21:46 10856人阅读 评论(0) 收藏 举报  分类: Qt(12)  版权声明:本文为博主原创文章,未经博主允许不得转载.   目录(?)[+]   一.简介     ...