题目描述
在成功地发明了魔方之后,拉比克先生发明了它的二维版本,称作魔板。这是一张有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”:魔板中央作顺时针旋转。
下面是对基本状态进行操作的示范:
8 7 6 5 4 1 2 3 1 7 2 4
A: 1 2 3 4 B: 5 8 7 6 C: 8 6 3 5
对于每种可能的状态,这三种基本操作都可以使用。
你要编程计算用最少的基本操作完成基本状态到特殊状态的转换,输出基本操作序列。
【输入】
只有一行,包括8个整数,用空格分开(这些整数在范围 1——8 之间),表示目标状态。
【输出】
Line 1:包括一个整数,表示最短操作序列的长度。
Line 2:在字典序中最早出现的操作序列,用字符串表示,除最后一行外,每行输出60个字符。
样例
in:
2 6 8 4 5 7 3 1
out:
7
BCABCCB
算法
bfs+经典的map记忆路径+string处理
代码
#include <bits/stdc++.h>
using namespace std;
typedef long long L;
const int N = 10;
string start, endd;
char g[2][N];
string te;
map<string, int> d;
map<string, pair<char,string >> mm;
queue<string> q;
void sset(string t) {
for (int i = 0, j = 0; j < 4; ++j, ++i) g[0][i] = t[j];
for (int j = 4,i = 3; j < 8; ++j,--i) g[1][i] = t[j];
}
string sget() {
string res;
for (int i = 0; i < 4; ++i) res += g[0][i];
for (int i = 3; i >= 0; --i) res += g[1][i];
return res;
}
string mv1(string t) {
sset(t);
for (int i = 0; i < 4; ++i) swap(g[1][i], g[0][i]);
return sget();
}
string mv2(string t) {
sset(t);
char l1 = g[0][3];
char l2 = g[1][3];
for (int i = 3; i >= 1; --i) {
g[0][i] = g[0][i - 1];
g[1][i] = g[1][i - 1];
}
g[0][0] = l1;
g[1][0] = l2;
return sget();
}
string mv3(string t) {
sset(t);
char c = g[0][1];
g[0][1] = g[1][1];
g[1][1] = g[1][2];
g[1][2] = g[0][2];
g[0][2] = c;
return sget();
}
void bfs() {
q.push(start);
while (q.size() > 0) {
string t = q.front();
q.pop();
string tt[3];
tt[0] = mv1(t);
tt[1] = mv2(t);
tt[2] = mv3(t);
sset(t);
char p = 'A'-1;
for (int i = 0; i < 3; ++i) {
p++;
if (d[tt[i]] > 0) continue;
d[tt[i]] = d[t] + 1;
mm[tt[i]] = {p,t};
if(tt[i] == endd||t==endd) return;
q.push(tt[i]);
}
}
}
int main() {
ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
char cc[8];
for (int i = 0; i < 8; ++i) {
cin >> cc[i];
endd += cc[i];
}
start = "12345678";
bfs();
cout << d[endd]<<endl;
string t = endd;
string ans;
while (t != start) {
ans = mm[t].first + ans;
t = mm[t].second;//前驱
}
cout << ans;
}