八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,左上角就是反着地

相关推荐
十八岁讨厌编程2 小时前
【算法训练营Day26】动态规划part2
算法·动态规划
智者知已应修善业3 小时前
【C++无数组矩阵对角线平均值保留2位小数】2022-11-18
c语言·c++·经验分享·笔记·算法·矩阵
papership3 小时前
【入门级-算法-6、排序算法: 计数排序】
数据结构·算法·排序算法
pengpeng023 小时前
力扣每日一题 611. 有效三角形的个数
算法·leetcode·职场和发展
2401_840105203 小时前
GESP C++5级 2025年6月编程2题解:最大公因数
数据结构·c++·算法
未知陨落4 小时前
LeetCode:56.子集
算法·leetcode·深度优先
PAK向日葵4 小时前
【算法导论】一道涉及到溢出处理的笔试题
算法·面试
哈泽尔都4 小时前
运动控制教学——5分钟学会样条曲线算法!(三次样条曲线,B样条曲线)
c++·人工智能·算法·机器学习·matlab·贪心算法·机器人
小镇学者4 小时前
【NOI】在信奥赛中 什么是函数交互题?
算法
未知陨落5 小时前
LeetCode:62.N皇后
算法·leetcode