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

相关推荐
大千AI助手3 小时前
DTW模版匹配:弹性对齐的时间序列相似度度量算法
人工智能·算法·机器学习·数据挖掘·模版匹配·dtw模版匹配
YuTaoShao5 小时前
【LeetCode 热题 100】48. 旋转图像——转置+水平翻转
java·算法·leetcode·职场和发展
生态遥感监测笔记5 小时前
GEE利用已有土地利用数据选取样本点并进行分类
人工智能·算法·机器学习·分类·数据挖掘
Tony沈哲6 小时前
macOS 上为 Compose Desktop 构建跨架构图像处理 dylib:OpenCV + libraw + libheif 实践指南
opencv·算法
刘海东刘海东6 小时前
结构型智能科技的关键可行性——信息型智能向结构型智能的转变(修改提纲)
人工智能·算法·机器学习
pumpkin845146 小时前
Rust 调用 C 函数的 FFI
c语言·算法·rust
挺菜的7 小时前
【算法刷题记录(简单题)003】统计大写字母个数(java代码实现)
java·数据结构·算法
mit6.8247 小时前
7.6 优先队列| dijkstra | hash | rust
算法
2401_858286117 小时前
125.【C语言】数据结构之归并排序递归解法
c语言·开发语言·数据结构·算法·排序算法·归并排序