QOJ 12255 - 36 Puzzle 题解

题意

给定一个 6×66\times 66×6 的网格图,每格分别不重复地填入了一个 1∼361\sim 361∼36 的数字。你每次可以选择一行或一列,对其向上或向下(选择列),向左或向右(选择行)循环位移,使得最终这个网格图是顺排的,即第 iii 行第 jjj 列的数是 (i−1)×n+j(i-1)\times n+j(i−1)×n+j。数据保证有解,构造出一种可行方案。本题题把 1∼361\sim 361∼36 映射到了 a∼z,0∼9\tt{a}\sim\tt{z},\tt{0}\sim{9}a∼z,0∼9,共 363636 个字符上。

例:

方法 1

发现要想交换 (x,y)(x,y)(x,y) 和 (x,y+1)(x,y+1)(x,y+1),可对第 xxx 行 / 第 yyy 列执行 ULURULURULURU 操作。要想交换 (x,y)(x,y)(x,y) 和 (x+1,y)(x+1,y)(x+1,y),可对第 xxx 行 / 第 yyy 列执行 LULDLULDLULDL 操作。实现邻项交换后,我们只要从上往下、从左往右地做就可以了。

cpp 复制代码
int to[200], id[7][7], a[7][7];
void init(){
    for (int i = 'a'; i <= 'z'; i++) to[i] = i - 'a' + 1;
    for (int i = '0'; i <= '9'; i++) to[i] = i - '0' + 27;
}
void f(int x, int y){
    string seq = "LULDLULDLULDL";
    for (char c : seq) cout << (c == 'L' || c == 'R' ? x : y) << " " << c << " 1\n";
}
void g(int x, int y){
    string seq = "ULURULURULURU";
    for (char c : seq) cout << (c == 'L' || c == 'R' ? x : y) << " " << c << " 1\n";
}
void solve(){
    init();
    int n = 6;
    for (int i = 1; i <= n; i++){
        for (int j = 1; j <= n; j++){
            char c; cin >> c;
            a[i][j] = to[c];
            id[i][j] = (i - 1) * n + j;
        }
    }
    for (int i = 1; i <= n; i++){
        for (int j = 1; j <= n; j++){
            int ni = -1, nj = -1;
            for (int x = 1; x <= n; x++) for (int y = 1; y <= n; y++) if (a[x][y] == id[i][j]) ni = x, nj = y;
            // 注意此处必须先调整列数,再调整行数
            while (nj > j){
                g(ni, nj - 1);
                swap(a[ni][nj - 1], a[ni][nj]);
                nj--;
            }
            while (nj < j){
                g(ni, nj);
                swap(a[ni][nj + 1], a[ni][nj]);
                nj++;
            }
            while (ni > i){
                f(ni - 1, nj);
                swap(a[ni][nj], a[ni - 1][nj]);
                ni--;
            }
        }
    }
}

方法 2(简述)

根据魔方公式 R,U,R′,U′R,U,R',U'R,U,R′,U′ 给我们的灵感,发现如果执行 RULD 可以交换 LLL 形的三个块,R..RUL..LD 等方式可以交换任意三个形成直角三角形的块。这样我们可以直接从上往下做,直到最后一行。发现选择三个东西做两次上述交换:

就可以实现同一行的三个东西轮换。然后我们发现长度为 666 的置换是奇置换,所以任意给定的网格图都是有解的。

相关推荐
浆果02078 分钟前
NanoTrack C++ — RK3588 实时目标跟踪
c++·目标跟踪·rk3588
Turbo正则21 分钟前
群论在AI中的应用概述
人工智能·算法·抽象代数
ysa05103025 分钟前
【并查集】判环
c++·笔记·算法
持力行35 分钟前
C/C++ 中的 char*:它标识数组吗?为什么能用下标访问?
c语言·c++
Jerry41 分钟前
KeetCode 44. 开发商购买土地
算法
Jerry1 小时前
KeetCode 58. 区间和
算法
Jerry2 小时前
LeetCode 209. 长度最小的子数组
算法
汉克老师2 小时前
GESP2026年6月认证C++六级( 第三部分编程题(2、满二叉树))精讲
c++·深度优先·树形dp·满二叉树·gesp六级·树形dfs
彦为君2 小时前
算法思维与经典智力题
java·前端·redis·算法
智能优化与强化学习2 小时前
Gym(Gymnasium)仿真环境详解(二):环境简介、入门算法、调参要点、核心挑战
算法·强化学习·gym·零基础入门·算法评估