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

相关推荐
CoovallyAIHub3 小时前
让本地知识引导AI追踪社区变迁,让AI真正理解社会现象
深度学习·算法·计算机视觉
CoderCodingNo3 小时前
【GESP】C++ 二级真题解析,[2025年12月]第一题环保能量球
开发语言·c++·算法
yumgpkpm3 小时前
预测:2026年大数据软件+AI大模型的发展趋势
大数据·人工智能·算法·zookeeper·kafka·开源·cloudera
CoovallyAIHub3 小时前
AAAI 2026这篇杰出论文说了什么?用LLM给CLIP换了个“聪明大脑”
深度学习·算法·计算机视觉
Physicist in Geophy.3 小时前
一维波动方程(从变分法角度)
线性代数·算法·机器学习
im_AMBER3 小时前
Leetcode 115 分割链表 | 随机链表的复制
数据结构·学习·算法·leetcode
Liue612312313 小时前
【YOLO11】基于C2CGA算法的金属零件涂胶缺陷检测与分类
人工智能·算法·分类
!!!!8133 小时前
蓝桥备赛Day1
数据结构·算法
Mr_Xuhhh3 小时前
介绍一下ref
开发语言·c++·算法
夏鹏今天学习了吗3 小时前
【LeetCode热题100(99/100)】柱状图中最大的矩形
算法·leetcode·职场和发展