news 2026/5/4 5:45:58

“N皇后”问题解法

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
“N皇后”问题解法

C++实现N皇后问题(回溯法详解+OJ适配)

一、核心问题分析

  • 不同行:由于每个皇后占一行,可简化为“逐行放置”(每行仅放一个皇后)

  • 不同列:同一列不能有两个皇后

  • 不同对角线:主对角线(x-y为常数)和副对角线(x+y为常数)不能有两个皇后

二、解题思路:回溯法

N皇后问题的核心解法是回溯法,本质是“尝试-回溯-再尝试”的暴力搜索优化思路:

  1. 初始化n×n的空棋盘(用'.'表示空位置);

  2. 逐行放置皇后:第i行放置一个皇后,标记其攻击范围(行、列、对角线);

  3. 递归处理下一行,重复步骤2;

  4. 若递归到第n行(所有皇后放置完成),则收集当前棋盘作为有效解;

  5. 回溯:撤销当前皇后的放置和攻击范围标记,尝试当前行的下一个列位置;

  6. 遍历所有可能的位置,直到收集完所有有效解。

关键优化:用“数字字符标记不同皇后的攻击范围”,确保回溯时能精准恢复棋盘状态,避免不同皇后的攻击范围混淆。

三、完整代码实现(OJ适配版)

以下代码已封装为LeetCode/OJ要求的Solution类,入口函数为solveNQueens(int n),可直接复制提交:

#include <iostream> #include <vector> #include <string> using namespace std; void conver(vector<vector<char>> a, vector<vector<string>>& b) { vector<string> temp; for ( auto char_row : a) { for (char& c : char_row) { if (c != 'Q' && c != '.') { c = '.'; } } string str(char_row.begin(), char_row.end()); temp.push_back(str); } b.push_back(temp); } void change(int x, int y, char c, vector<vector<char>>& a,int n) { for (int i = 0; i < n; i++) { if (a[x][i] == '.') a[x][i] = c; if(a[i][y]=='.') a[i][y] = c; } for (int i = -(n - 1); i <= n - 1; i++) { if (x + i < n && x + i >= 0 && y + i >= 0 && y + i < n&&a[x+i][y+i]=='.') a[x + i][y + i] = c; if (x - i < n && x - i >= 0 && y + i < n && y + i >= 0&&a[x-i][y+i]=='.') a[x - i][y + i] = c; } } void restore(char c, int x, int y,int n,vector<vector<char>>&a) { for (int i = 0; i < n; i++) { for (int j = 0; j < n; j++) { if (a[i][j] == c) { a[i][j] = '.'; } } } } void queen( int num, int n, vector<vector<char>>& a, vector<vector<string>>& b) { if (num == n) { conver(a, b); return; } for (int i= 0; i < n; i++) { if (a[num][i] == '.') { change(num, i, '0'+num, a, n); a[num][i] = 'Q'; queen(num + 1, n, a, b); restore('0'+num, num, i, n, a); a[num][i] = '.'; } } return; } int main() { int n = 4; vector<vector<char>> a(n, vector<char>(n)); vector<vector<string>> b; for (int i = 0; i < n; i++) { for (int j = 0; j < n; j++) { a[i][j] = '.'; } } int num = 0; queen(num , n, a, b); for (const auto& str_arr : b) { for (const auto& str : str_arr) { cout << str<<endl; } cout <<endl<<"---------------"<< endl; } }

四、核心代码模块解析

1. 主函数:int main()

作用:OJ的统一入口,负责初始化棋盘、调用递归函数、返回最终结果。

  • 初始化n×n的棋盘a,所有位置设为'.'(空);

  • 创建b存储所有有效解法(二维字符串数组,每个子数组是一个完整的棋盘);

  • 调用核心递归函数queen,从第0行(num=0)开始放置皇后。

2. 核心递归函数:queen(int num, int n, vector<vector<char>>& a, vector<vector<string>>& b)

作用:实现回溯法的核心逻辑——逐行放置皇后、递归、回溯。

  • 终止条件num == n,表示已处理完第0~n-1行(所有皇后放置完成),调用conver转换结果并收集;

  • 逐行遍历num表示当前处理的行,遍历当前行的所有列(i从0到n-1);

  • 放置皇后:若当前位置a[num][i] == '.'(空),则:

    • '0' + num生成当前皇后的专属标记(如第0行皇后用'0',第1行用'1');

    • 调用change标记攻击范围;

    • 将当前位置设为'Q'(放置皇后);

    • 递归处理下一行(num+1);

    • 回溯:调用restore恢复攻击范围标记,将当前位置设回'.'(撤销皇后)。

3. 攻击范围标记:change(int x, int y, char c, vector<vector<char>>& a, int n)

作用:标记皇后(x,y)的攻击范围(行、列、主对角线、副对角线),用字符c(专属标记)标记,避免与其他皇后混淆。

  • 标记当前行:遍历第x行所有列,空位置设为c

  • 标记当前列:遍历第y列所有行,空位置设为c

  • 标记主对角线:x+i, y+i(x-y为常数),需判断边界(0≤nx<n,0≤ny<n);

  • 标记副对角线:x-i, y+i(x+y为常数),同样判断边界。

4. 回溯恢复:restore(char c, int x, int y, int n, vector<vector<char>>& a)

作用:撤销当前皇后的攻击范围标记——将所有标记为c的位置恢复为'.',确保回溯后棋盘状态正确。

关键:由于每个皇后的标记c是唯一的('0'~'n-1'),恢复时不会影响其他皇后的标记。

5. 结果转换:conver(vector<vector<char>> a, vector<vector<string>>& b)

作用:将标记后的棋盘转换为OJ要求的输出格式——过滤攻击范围标记('0'~'n-1'),仅保留'Q'(皇后)和'.'(空)。

注意:参数a采用值传递,避免修改原棋盘的状态(不影响后续回溯)。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/30 7:34:59

Part 11|模块划分并非越细越好,关键在于明确职责边界

在明确要从业务边界开始拆系统之后&#xff0c;我很快遇到了一个新的现实问题&#xff1a;业务边界清楚了&#xff0c;但模块到底要拆到什么程度&#xff1f;一开始&#xff0c;我其实很容易走向一个极端&#xff1a; 既然要清晰&#xff0c;那就尽量拆细一点。 但真正把模块往…

作者头像 李华
网站建设 2026/4/30 23:46:41

设计模式之-观察者模式

1.先来看一个简单的例子 // 观察者 class Observer {update(data){// 观察者收到数据变化&#xff0c;自行处理要做的事情console.log(接收到了数据&#xff1a;--,data);} } // 目标 class Subject {constructor(){// 维护所有的观察者列表this.observers [];}add(ob){// 添…

作者头像 李华
网站建设 2026/5/4 5:45:53

单北斗GNSS在桥梁形变监测与维护中的应用与优势分析

本文旨在深度分析单北斗GNSS在桥梁形变监测与维护中的应用与优势。首先&#xff0c;单北斗GNSS厂家提供的各类产品&#xff0c;如变形监测一体机和传感器&#xff0c;具备独特的技术特性&#xff0c;能够满足不同桥梁监测需求。其次&#xff0c;监测系统的定制与实施方法&#…

作者头像 李华
网站建设 2026/5/3 20:56:41

Arbess从基础到实践(16) - 集成GitHub实现Java项目构建并自动化Docker部署

Arbess 是一款国产开源免费的 CI/CD 工具&#xff0c;支持免费自动化部署&#xff0c;一键安装零配置。本文将详细介绍如何安装并使用ArbessGitHub实现Docker项目自动化构建部署 1、GitHub 配置 本章节将介绍如何创建GitHub个人访问令牌&#xff0c;提供给Arbess克隆源码。 …

作者头像 李华
网站建设 2026/5/2 8:11:09

基于Python的健身房管理系统源码设计与文档

前言在健身房精细化运营需求提升、传统管理模式存在 “会员管理混乱、课程预约低效、数据统计滞后、私教跟进缺位” 的痛点背景下&#xff0c;基于 Python 的健身房管理系统构建具有重要的商业与实用价值&#xff1a;从会员管理层面&#xff0c;系统依托 Python 的数据库交互能…

作者头像 李华
网站建设 2026/4/30 11:16:42

NVIDIA HGX™ B300 GPU Droplet 服务器,即将上线DigitalOcean 云平台!

人工智能正以史无前例的速度演进&#xff0c;新的模型和繁重的负载不断突破可能的边界。从复杂的大型语言模型&#xff08;LLM&#xff09;到精密的科学模拟&#xff0c;开发者与企业都需要获得最强大、最高效的算力基础设施。在 DigitalOcean&#xff0c;我们致力于提供顶级的…

作者头像 李华