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

相关推荐
一只叁木Meow18 小时前
电商 SKU 选择器:用算法实现优雅的用户交互
前端·javascript·算法
代码中介商18 小时前
红黑树完全指南:从五条性质到完整插入删除实现
数据结构·算法
JieE21218 小时前
反转链表:从双指针到递归,吃透链表反转的核心逻辑
javascript·算法
代码村新手18 小时前
C++-多态
开发语言·c++
玖釉-19 小时前
旋转图像:从矩阵转置、镜像到坐标变换的系统理解
c++·windows·算法·图形渲染
fengenrong19 小时前
20260522
算法
一条大祥脚19 小时前
Codeforces Round 1099 (Div. 2) 构造|贪心|图论|还原数组
java·算法·图论
咩咦19 小时前
C++学习笔记23:const 成员函数
c++·学习笔记·类和对象·const·this指针·const成员函数
Sheldon Chao20 小时前
Lecture 7 基于策略梯度的算法
人工智能·算法·机器学习
始三角龙20 小时前
LeetCode hoot 100 -- 缺失的第一个正整数
算法·leetcode·职场和发展