八N皇后问题

1 问题的提出

在8X8格的国际象棋上摆放八个皇后,使其不能互相攻击,即任意两个皇后都不能处于同一行、同一列或同一斜线上,问有多少种摆法

我们的任务就是用MATLAB进行求解

2 数学模型的构建

首先我们分析题目就是

任意两个皇后都不能处于同一行、同一列或同一斜线上

复制代码
global board sum;
board = zeros(8,8);
sum = 0;
dfs(1);
disp(sum);

function dfs(x)
    global board sum;
    if x > 8
        sum = sum + 1;
        return;
    end

    for i = 1:8
        if issafe(x,i,board)
            board(x,i) = 1;
            dfs(x+1);
            board(x,i) = 0;
        end
    end
end

function safe = issafe(row, col, board)
    safe = true;
    
    % 检查列
    for i = 1:row-1
        if board(i, col) == 1
            safe = false;
            return;
        end
    end

    % 检查右上角
    i = row - 1;
    j = col + 1;
    while i >= 1 && j <= 8
        if board(i, j) == 1
            safe = false;
            return;
        end
        i = i - 1;
        j = j + 1;
    end

    % 检查左上角
    i = row - 1;
    j = col - 1;
    while i >= 1 && j >= 1
        if board(i, j) == 1
            safe = false;
            return;
        end
        i = i - 1;
        j = j - 1;
    end
end

我们要学习这里面的思想

3 模块1 dfs搜索函数

复制代码
function dfs(x)
    global board sum;
    if x > 8
        sum = sum + 1;
        return;
    end

    for i = 1:8
        if issafe(x,i,board)
            board(x,i) = 1;
            dfs(x+1);
            board(x,i) = 0;
        end
    end
end

我们有8个皇后,那就是for循环循环8行,每次放置一个棋子就进行一次判断,然后判断这个棋子可不可以落在这里如过可以那么久进入到下一行,x进行+1秒如果不可以的话,那么就进入这一行的下一个格子下一个,这就是枚举8行8列

当这个x > 8的话,那么就是放置成功了

4 检查模块

复制代码
function safe = issafe(row, col, board)
    safe = true;
    
    % 检查列
    for i = 1:row-1
        if board(i, col) == 1
            safe = false;
            return;
        end
    end

    % 检查右上角
    i = row - 1;
    j = col + 1;
    while i >= 1 && j <= 8
        if board(i, j) == 1
            safe = false;
            return;
        end
        i = i - 1;
        j = j + 1;
    end

    % 检查左上角
    i = row - 1;
    j = col - 1;
    while i >= 1 && j >= 1
        if board(i, j) == 1
            safe = false;
            return;
        end
        i = i - 1;
        j = j - 1;
    end
end

首先我们的行是已经操作完的了,就是在判断行的话是在递归的过程中进行讨论的,然后就是只需要判断这个列是否成立就好了,然后斜边的话,那不就是直接判断左上角和右上角就好了

左上角和右上角的检查

复制代码
    % 检查右上角
    i = row - 1;
    j = col + 1;
    while i >= 1 && j <= 8
        if board(i, j) == 1
            safe = false;
            return;
        end
        i = i - 1;
        j = j + 1;
    end

我们知道右上角就是不断的进行加1嘛,这个行的话,列就是不断地进行减1,左上角就是反着地

相关推荐
心中有国也有家1 小时前
cann-recipes-infer:昇腾 NPU 推理的“菜谱集合”
经验分享·笔记·学习·算法
绝知此事1 小时前
【算法突围 01】线性结构与哈希表:后端开发的收纳术
java·数据结构·算法·面试·jdk·散列表
碧海银沙音频科技研究院1 小时前
通话AEC与语音识别AEC的软硬回采链路
深度学习·算法·语音识别
csdn_aspnet2 小时前
Python 算法快闪 LeetCode 编号 70 - 爬楼梯
python·算法·leetcode·职场和发展
m0_629494735 小时前
LeetCode 热题 100-----26.环形链表 II
数据结构·算法·leetcode·链表
壹号用户5 小时前
用队列实现栈
数据结构·算法
做人求其滴5 小时前
面试经典 150 题 380 274
c++·算法·面试·职场和发展·力扣
daad7775 小时前
记一组无人机IMU传感器数据
算法
计算机安禾5 小时前
【c++面向对象编程】第42篇:模板特化与偏特化:为特定类型定制实现
开发语言·c++·算法