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 的置换是奇置换,所以任意给定的网格图都是有解的。

相关推荐
AI进化营-智能译站1 小时前
ROS2 C++开发系列13-运算符重载让ROS2消息处理更自然
java·开发语言·c++·ai
AI科技星1 小时前
《全域数学》第一部 数术本源 第三卷 代数原本第14篇 附录二 猜想证明【乖乖数学】
人工智能·算法·数学建模·数据挖掘·量子计算
Wect2 小时前
LeetCode 72. 编辑距离:动态规划经典题解
前端·算法·typescript
zhouwy1132 小时前
Poco 与 libevent 网络编程
c++
憨波个2 小时前
【说话人日志】DOVER-Lap:overlap-aware diarization 输出融合算法
人工智能·深度学习·算法·音频·语音识别
叼烟扛炮2 小时前
C++第四讲:类和对象(下)
c++·算法·类和对象
Rabitebla2 小时前
vector 的骨架:三根指针、模板陷阱与迭代器失效的第一现场
开发语言·数据结构·c++·算法
代码不停2 小时前
BFS解决floodfill算法题目练习
算法·宽度优先
上弦月-编程2 小时前
C语言指针从入门到实战
java·jvm·算法