魔板 (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 ...
随机推荐
- linux软件操作
操作 命令 ubuntu 源操作 源配置 https://www.cnblogs.com/wenlin-gk/p/11146228.html 源更新 sudo apt-get update 查看源中包 ...
- 043:Django使用原生SQL语句操作数据库
Django使用原生SQL语句操作数据库 Django配置连接数据库: 在操作数据库之前,首先先要连接数据库.这里我们以配置 MySQL 为例来讲解. Django 连接数据库,不需要单独的创建一个连 ...
- Centos 6.4 x86_64 最小化安装后的优化——还需要整理
Centos 6.4 x86_64 最小化安装后的优化 购买了服务器以后要做的第一件事就是安装操作系统了,这里推荐安装 Centos 6.4 x86_64,安装系统时要选择最小化安装(不需要图 ...
- 【leetcode】473. Matchsticks to Square
题目如下: 解题思路:居然把卖火柴的小女孩都搬出来了.题目的意思是输入一个数组,判断能否把数组分成四个子数组,使得每个子数组的和相等.首先我们可以很容易的求出每个子数组的和应该是avg = sum(n ...
- 8:Spring Boot中thymeleaf模板中使用 Shiro标签
1,添加 pom.xml grade: compile('com.github.theborakompanioni:thymeleaf-extras-shiro:1.2.1') 2, Subject ...
- Vue中v-for配合使用Swiper插件问题
问题描述: 在一个页面中需要一个用swiper的轮播图,数据大概有40条,每一屏幕的swiper只显示其中的n条数据. 代码描述: <div id="app"> < ...
- Eclipse中jar包的导出与导入
JAR的含义: JAR是Java的档案文件,是Java Archive File的缩写.jar文件是一种压缩文件,就是以特定类型压缩包的形式存在的完整Java项目.通常通过导入jar包的方式来使用实现 ...
- Handling Configuration Changes with Fragments
This post addresses a common question that is frequently asked on StackOverflow: What is the best wa ...
- Java网络编程与NIO详解11:Tomcat中的Connector源码分析(NIO)
本文转载 https://www.javadoop.com 本系列文章将整理到我在GitHub上的<Java面试指南>仓库,更多精彩内容请到我的仓库里查看 https://github.c ...
- ODB Examples
http://www.codesynthesis.com/products/odb/examples.xhtml The following list gives an overview of the ...