复试上机准备之马跳棋

Author Avatar
patrickcty 3月 06, 2019

题目要求

假设国际象棋棋盘有 5*5 共 25 个格子。设计一个程序,使棋子从初始位置(棋盘编号 为 1 的位)开始跳马,能够把棋盘的格子全部都走一遍,每个格子只允许走一次。

(1)输出一个如图 2 的解,左上角为第一步起点。

(2)总共有多少解。

P.S 国际象棋的棋子是在格子中间的。国际象棋中的“马走日”,如下图所示,第一步为[1,1], 第二步为[2,8]或[2,12],第三步可以是[3,5]或[3,21]等,以此类推。

分析

这是一个常规的回溯题,要注意的就是一共有八个走的方向,在返回的时候记得还原原本的状态值。

一共有 304 解,代码如下:

# include <iostream>

using namespace std;

int arr[5][5] = {0};
int vis_num = 1;
int cnt = 0;

void horse_jump(int x, int y) {
    // 输出走法
    if (vis_num == 25) {
        for (int i = 0; i < 5; ++i) {
            for (int j = 0; j < 5; ++j) {
                cout << arr[i][j] << ' ';
            }
            cout << endl;
        }
        cout << endl;
        cnt++;
    }

    // 右二下一
    if (x + 2 < 5 && y + 1 < 5 && arr[x + 2][y + 1] == 0) {
        arr[x + 2][y + 1] = ++vis_num;
        horse_jump(x + 2, y + 1);
        // 重置状态
        arr[x + 2][y + 1] = 0;
        --vis_num;
    }
    // 右一下二
    if (x + 1 < 5 && y + 2 < 5 && arr[x + 1][y + 2] == 0) {
        arr[x + 1][y + 2] = ++vis_num;
        horse_jump(x + 1, y + 2);
        // 重置状态
        arr[x + 1][y + 2] = 0;
        --vis_num;
    }
    // 左二上一
    if (x - 2 >= 0 && y - 1 >= 0 && arr[x - 2][y - 1] == 0) {
        arr[x - 2][y - 1] = ++vis_num;
        horse_jump(x - 2, y - 1);
        // 重置状态
        arr[x - 2][y - 1] = 0;
        --vis_num;
    }
    // 左一上二
    if (x - 1 >= 0 && y - 2 >= 0 && arr[x - 1][y - 2] == 0) {
        arr[x - 1][y - 2] = ++vis_num;
        horse_jump(x - 1, y - 2);
        // 重置状态
        arr[x - 1][y - 2] = 0;
        --vis_num;
    }
    // 左二下一
    if (x - 2 >= 0 && y + 1 < 5 && arr[x - 2][y + 1] == 0) {
        arr[x - 2][y + 1] = ++vis_num;
        horse_jump(x - 2, y + 1);
        // 重置状态
        arr[x - 2][y + 1] = 0;
        --vis_num;
    }
    // 左一下二
    if (x - 1 >= 0 && y + 2 < 5 && arr[x - 1][y + 2] == 0) {
        arr[x - 1][y + 2] = ++vis_num;
        horse_jump(x - 1, y + 2);
        // 重置状态
        arr[x - 1][y + 2] = 0;
        --vis_num;
    }
    // 右二上一
    if (x + 2 < 5 && y - 1 >= 0 && arr[x + 2][y - 1] == 0) {
        arr[x + 2][y - 1] = ++vis_num;
        horse_jump(x + 2, y - 1);
        // 重置状态
        arr[x + 2][y - 1] = 0;
        --vis_num;
    }
    // 右一上二
    if (x + 1 < 5 && y - 2 >= 0 && arr[x + 1][y - 2] == 0) {
        arr[x + 1][y - 2] = ++vis_num;
        horse_jump(x + 1, y - 2);
        // 重置状态
        arr[x + 1][y - 2] = 0;
        --vis_num;
    }
}


int main() {
    arr[0][0] = 1;
    horse_jump(0, 0);
    cout << cnt << endl;

    return 0;
}