(新手向)N皇后问题详解(DFS算法)

article/2025/10/6 0:56:25

非常经典的一道题:

N皇后问题: 
国际象棋中皇后的势力范围覆盖其所在的行、列以及两条对角线,现在考察如下问题:如何在n x n的棋盘上放置n个皇后,使得她们彼此互不攻击 。

免去麻烦我们这里假定n不是很大。。

(图片来自百度百科(这是8皇后问题的一种解法))

某leetcode大犇曾说过:“这个问题和解数独题目有一个很大的共同点,那就是:我都不会。”

好了下面开始分析:(废话警告)

初步判断这问题的特点有:

1.有个场地来放置单位。

2.各个单位之间有制约。

3.没有特殊的数学方法,得把某一个摆法摆出来才能判断是否可行。

于是萌新一般都会这么想:对于1,:我搞个二维数组来存。对于2:我搞个判定函数来一个一个判定。对于3:我暴力枚举。

那么算法框架大概就是:我对二维数组的所有情况进行枚举,然后对每种情况进行判定

over ,输个数字n,按下回车,双手离开键盘,等了老半天发现命令提示符只有个光标在跳动 |

如果是在做网题,说不定就判定是超时或者内存溢出,过不了。

然后就开始思考怎么优化:

对于1,我可以缩减吗····,改成一维数组,,,似乎徒增麻烦。

对于2,我可以有更巧妙的判断方法吗?·············有个鬼,还不得把每个棋子的每一行每一列每两个斜线都瞅一下。

对于3,我一定要把所有情况都举出来嘛?···(于是脑子里面开始摆起了棋子,模拟算法过程)

然后就会发现  比如第一行前两个格子一开始就摆了两个相邻的棋子,诶这不明摆着皇后互怼了吗!后面还继续枚举就太傻了吧···

这种傻事情做了不是白白增加耗时吗!

所以就思考如何在这种“傻情况”出现的时候就pass掉后面所有情况。。。

于是脑子陷入了一团乱麻。。。

打住!

咱们换个思考方式,既然直接想有点困难,不如我们想办法先做点处理,让这些玩意儿便于我们把弄。

再回头想想规则,皇后可以吃掉所处的一行一列以及斜线上的棋子,那也就是说,每一行都最多只能有一个棋子,每一列都最多只能有一个棋子,

那我们不妨把每一行看成一组!

那么这一组就只有n个可能性(一行n个位置 每一次只有一个位置被占用)

也就是单个一行有n种可能性

因为有n行

哈哈那么就瞬间只需要讨论n*n种可能性了!

 还记得我们之前的思考吗?:

对于3,我一定要把所有情况都举出来嘛?···(于是脑子里面开始摆起了棋子,模拟算法过程)然后就会发现  比如第一行前两个格子一开始就摆了两个相邻的棋子,诶这不明摆着皇后互怼了吗!后面还继续枚举就太傻了吧···这种傻事情做了不是白白增加耗时吗!所以就思考如何在这种“傻情况”出现的时候就pass掉后面所有情况。。。

像这种情况:

(我的天这皇后画的...)

显然下面3,4,5····行都不用枚举了,直接pass掉

于是我们思路慢慢清晰了起来:

我们从第一行开始枚举,第一行第一格:

然后枚举第二行,也从第一格开始:

 判定一下,哇,不行,咋办呢?这情况下面所有行都没有枚举的必要了,但是本行还是可以继续的。。。于是我们开始枚举第二行第二个情况:

不行(斜线上互吃)。

继续第二行下一个情况:

行嘞!

那么我们就可以开第三行了:

不行(一列上吃)。

不行(斜线上吃)

 不行。

 还不行

 行嘞!

··················

好了,算法思路大概就出来了。

 我们就这样干下去,直到最后一行可行,那么我们就获得了一个可行解了,

然后我们想继续获得其他解,那么继续枚举下去,但是要退一步:

先从倒数第一行开始,我们枚举下一个,有解则输出,无解则继续。

倒数第一行结束了,咋办?

别忘了倒数第二行的情况还不一定枚举完了呢。

于是退到倒数第二行,继续如上操作。

··················


 干说着没用,撸代码:

#include<iostream>using namespace std;
int main()
{return 0;
}

先准备好需要用的存储(这里用64个绰绰有余)

ans是用来存每一次的解 ans[1]表示第一行的那个元素的所在列的位置

#include<iostream>using namespace std;
bool colMark[64] = { 0 };//元素所在列  如果有了元素在第k列 那么colMark[k]就标记上true
bool naMark[64] = { 0 };//捺,撇···顾名思义,表示元素所在的两个斜线o.o 原理同上
bool pieMark[64] = { 0 };
int total = 0;           //解的总个数
int N = 0;
int ans[64] = { 0 };
int main()
{return 0;
}

然后下面先写一个显示函数: 

void showOneSolution()//用来显示一个解
{total++;for (int i = 1; i<=N; ++i){cout << ans[i] << " ";}cout << endl;
}

下面是核心:

void Dfs(int i)   //可以百度学习 深度优先搜索
{if (i > N)              //判断是否已经枚举完了N行 {showOneSolution();  //枚举完了就输出(此刻我们处于N+1行)return;             //返回到第N行}for (int j = 1; j <= N; ++j) //这里的j表示  本行的第j列{if ((!colMark[j])&&(!naMark[j-i+N])&&(!pieMark[i+j]))    //这里的"j-i+N"  "i+j"建议自己体会{          //判断这个格子所在的两个斜线和所在列是否为空colMark[j] = true;naMark[j - i + N] = true;pieMark[i + j] = true;ans[i] = j;Dfs(i + 1);colMark[j] = false;      //这个格子枚举完了要进入本行第j+1个格子(或者结束本行) 走之前别忘了恢复标记naMark[j - i + N] = false;pieMark[i + j] = false;}}
}

 最后组装起来:(当然啦,还是建议自己撸代码,你会发现很多意想不到的问题~解决问题的同时也是进步)

#include<iostream>using namespace std;
bool colMark[64] = { 0 };
bool naMark[64] = { 0 };
bool pieMark[64] = { 0 };
int total = 0;
int N = 0;
int ans[64] = { 0 };
void showOneSolution()//用来显示一个解
{total++;for (int i = 1; i<=N; ++i){cout << ans[i] << " ";}cout << endl;
}
void Dfs(int i)
{if (i > N)              //判断是否已经枚举完了N行{showOneSolution();  //枚举完了就输出(此刻我们处于N+1行)return;             //返回到第N行}for (int j = 1; j <= N; ++j) //这里的j表示  本行的第j列{if ((!colMark[j]) && (!naMark[j - i + N]) && (!pieMark[i + j])){          //判断这个格子所在的两个斜线和所在列是否为空colMark[j] = true;naMark[j - i + N] = true;pieMark[i + j] = true;ans[i] = j;Dfs(i + 1);colMark[j] = false;      // 走之前别忘了恢复标记naMark[j - i + N] = false;pieMark[i + j] = false;}}
}
int main()
{cin >> N;Dfs(1);cout << total;system("pause");return 0;
}

总结:

思路混乱不妨换个角度理一理。

在纸上面画一画算法过程会有不少帮助。

代码少出错还是得多撸。

想什么呢?!撸代码!


http://chatgpt.dhexx.cn/article/FsCyhi1o.shtml

相关文章

n皇后问题-c语言实现

n皇后问题描述为:在一个nxn的棋盘上摆放n个皇后&#xff0c;要求任意两个皇后不能冲突&#xff0c;即任意两个皇后不在同一行、同一列或者同一斜线上。 1,1 1,2 1,3 1,4 2,1 2,2 2,3 2,4 3,1 3,2 3,3 3,4 4,1 4,2 4,3 4,4 上面是4皇后摆放方案&#xff0c;只有…

n皇后问题合集

八皇后问题是dfs的经典问题&#xff0c;目前遇到过的类型大致有这么几种&#xff1a; 1.经典n皇后问题 题目描述 n−皇后问题是指将 n 个皇后放在 nn 的国际象棋棋盘上&#xff0c;使得皇后不能相互攻击到&#xff0c;即任意两个皇后都不能处于同一行、同一列或同一斜线上。 …

N皇后问题(c语言实现)

问题描述&#xff1a; 有一个n*n的棋盘&#xff0c;在这个棋盘中放n个皇后&#xff0c;使得这n个皇后&#xff0c;任意两个皇后不在同一行&#xff0c;同一列&#xff0c;同一条对角线。例如&#xff0c;当n等于4时&#xff0c;有两种摆法。 输入只有一个整数n。 思路 如果…

N皇后问题

问题描述 n 皇后问题研究的是如何将 n 个皇后放置在 nn 的棋盘上&#xff0c;并且使皇后彼此之间不能相互攻击。 上图为 8 皇后问题的一种解法。 给定一个整数 n&#xff0c;返回所有不同的 n 皇后问题的解决方案。 每一种解法包含一个明确的 n 皇后问题的棋子放置方案&#…

回溯算法之N皇后问题

问题描述 什么是皇后问题(有一定了解可以直接跳过这个部分看求解部分哦) 八皇后问题&#xff08;英文&#xff1a;Eight queens&#xff09;&#xff0c;是由国际西洋棋棋手马克斯贝瑟尔于1848年提出的问题&#xff0c;是回溯算法的典型案例。 问题表述为&#xff1a;在88格的…

N皇后问题(分支限界法)

问题描述&#xff1a; 在 n * n 格的棋盘上放置彼此不受攻击的 n 个皇后。按照国际象棋的规则&#xff0c;皇后可以攻击与之处在同一行或同一列或同一斜线上的棋子。 n 皇后问题等价于在 n * n 的棋盘上放置 n 个皇后&#xff0c;任何 2个皇后不放在同一行或同一列或同一斜线上…

N皇后问题详解

文章目录 一、题目描述二、题目解析&#xff08;1&#xff09;思考一(集合回溯)&#xff08;2&#xff09;思考二(数组深度递归)&#xff08;3&#xff09;思考三(位运算) 一、题目描述 N 皇后问题是指在 n * n 的棋盘上要摆 n 个皇后&#xff0c; 要求&#xff1a;任何两个皇…

3.2 回溯法—N皇后问题

1. 问题描述 在 n n n\times n nn的棋盘上摆放 n n n个皇后&#xff0c;使任意两个皇后都不能处于同一行、同一列或同一斜线上 2. 问题分析 下以求解4皇后问题为例&#xff0c;分析4皇后问题的排列树以及回溯过程&#xff1a; 搜索及回溯过程&#xff1a; 解空间树&#…

【经典算法】N皇后问题

✨前言✨ N皇后问题经典的解决方案是暴力递归&#xff0c;其时间复杂度是O(2^n),因此常用来测试计算机的算力。今天我会给大家带来经典方法的详解&#xff0c;也会给大家展示N皇后优化后的大神解法。做一道经典题目&#xff0c;来一场思维旅行。 目录 ✨前言✨ &#x1f4a1;…

n皇后问题(回溯法)(超易理解)(详细注释)

问题&#xff1a; N皇后问题是一个经典的问题&#xff0c;描述如下&#xff1a; 在的方格棋盘需要放置N个皇后&#xff0c;使得它们不相互攻击&#xff0c; 即任意2个皇后不允许处在同一排&#xff0c;同一列&#xff0c;也不允许处在与棋盘边框成45度角的斜线上。 对于给定的N…

n皇后问题(回溯法)

目录 1.问题描述 2.问题分析 3.完整源码 1.问题描述 八皇后问题是十九世纪著名的数学家高斯于1850年提出的。问题是:在88的棋盘上摆放八个皇后&#xff0c;使其不能互相攻击&#xff0c;即任意两个皇后都不能处于同一行、同一列或同一斜线上。可以把八皇后问题扩展到n皇后问…

【浏览器跨域】跨域访问接口浏览器怎么设置?

window系统 1、右键浏览器&#xff0c;点击属性 2、找到目标&#xff0c;目标输入框内容最后面加上 --disable-web-security --user-data-dirC:\MyChromeDevUserData&#xff0c;–user-data-dir 3、点击确定 4、重新打开浏览器即可 mac系统 终端输入&#xff0c;重新打开浏…

AJAX跨域访问(不同域之间相互访问)

目录 一、跨域&#xff1a;二、同源策略&#xff1a;三、解决Ajax跨域问题的方案&#xff1a;方案一&#xff1a;设置响应头方案二&#xff1a;jsonp方案三&#xff1a;jQuery封装jsonp方案四&#xff1a;代理机制&#xff08;httpclient&#xff09;方案五&#xff1a;nginx反…

nginx配置图片跨域访问

在server段中添加红框内的图片跨域内容 参数 location ~* .*.(gif|jpg|jpeg|png|bmp|swf)$ { add_header Access-Control-Allow-Origin *; add_header Access-Control-Allow-Headers X-Requested-With; add_header Access-Control-Allow-Methods GET,POST,PUT,DELETE,OPTIONS…

浏览器跨域访问限制

一.什么是跨域 广义的跨域&#xff1a; (1) 资源跳转&#xff1a;A链接、重定向、表单提交 (2) 资源嵌入&#xff1a;<link>、<script>、<img>、<frame>等dom标签&#xff0c;还有样式中background:url()、font-face()等文件外链 (3) 脚本请求&#x…

Django 多方式实现跨域访问

一、什么是跨域 1.1 跨越介绍 跨域&#xff0c;是指浏览器不能执行其他网站的脚本。它是由浏览器的同源策略造成的&#xff0c;是浏览器对JavaScript实施的安全限制。 这里说明一下&#xff0c;无法跨域是浏览器对于用户安全的考虑&#xff0c;如果自己写个没有同源策略的浏览…

Flask中跨域访问的实现

在我们访问不同的服务器时&#xff0c;就会涉及到了跨域的问题。因为不同域名之间是无法进行交流的&#xff0c;然后跨域就打破了这种规则的限制。说起Flask中的跨域&#xff0c;就不得不提到CORS组件了&#xff0c;相信大家在其它框架中也见过了它的身影。下面我们就跨域问题和…

Tomcat设置允许跨域访问

开发React项目时前端通过axios向后端代码发起请求调试的时候由于后端代码运行在8080端口而React项目运行在3000端口导致浏览器的同源策略禁止跨域请求&#xff0c;因此需修改Tomcat配置文件web.xml以开放跨域访问。 在tomcat的web.xml文件末尾加上&#xff1a; <filter>…

express 允许跨域访问

设置Express const express require(express) const bodyParser require(body-parser)const app express() // 处理参数 app.use(bodyParser.json()); app.use(bodyParser.urlencoded({ extended: false }));// 设置允许跨域访问该服务 app.all(*, function (req, res, nex…

tomcat 设置允许跨域访问

既然想到使用tomcat进行跨域的设置&#xff0c;而不使用在项目中设置header来解决&#xff0c;说明你也是tomcat 下的资源需要做跨域处理吧&#xff1f;这也是一个统一的允许跨域设置&#xff0c;tomcat下的所有请求都将放开&#xff0c;请注意。 具体步骤&#xff1a; 1.首先…