news 2026/4/18 3:57:21

“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/18 4:19:10

【编号645】全国省市县行政区划矢量数据2025年更新

今天小编整理分享的是 全国省市县行政区划矢量数据2025年更新 。市边界省边界县边界概况数据概况全国省市县行政区划矢量数据2025年更新全国省市县行政区划矢量数据2025年更新。shp/geojson数据&#xff0c;WGS84坐标系。包括我国省份、地级市、区县三个层级的行政区划矢量数…

作者头像 李华
网站建设 2026/4/15 13:50:57

TypeScript语法

这些是 MobX-State-Tree (MST)​ 的核心语法&#xff0c;用于定义可观察的状态树模型。我来详细解释每个部分&#xff1a;1. types对象的作用types是 MST 库导出的类型构建器集合&#xff0c;用于定义数据模型的形状、验证规则和行为。import { types } from "mobx-state-…

作者头像 李华
网站建设 2026/4/16 23:50:32

基于 RT-Thread Studio 实战:ESP8266+MQTT

作为嵌入式开发爱好者&#xff0c;在完成 ESP8266 结合 MQTT 协议的物联网通信实战后&#xff0c;我决定将整个过程记录并分享出来。本文以正点原子 F429 开发板 ESP8266 无线模块为硬件载体&#xff0c;基于 RT-Thread Studio 开发环境&#xff0c;搭配 EMQX 云服务器&#x…

作者头像 李华
网站建设 2026/4/18 2:56:35

智能压力测试代理系统:基于AI的自动化压测解决方案

作者&#xff1a;质立方qiyanfei 原创文章&#xff0c;转载请注明出处项目概述 实现了一个功能强大的智能压力测试代理系统。该系统基于LangGraph框架和DeepSeek AI模型&#xff0c;通过多智能体协作实现了全流程的自动化压力测试。 系统架构设计 1. 核心组件架构 用户请求 → …

作者头像 李华
网站建设 2026/3/27 17:04:39

​OOTD新硬核!ROG幻X 2025,解锁科技博主的潮流穿搭与全能生产。

对于每天穿梭于秀场、片场和咖啡馆的时尚博主来说&#xff0c;选择笔记本的要求向来苛刻。既要能装在迷你包里随拍随修&#xff0c;又要撑得起4K素材剪辑和AI修图的性能需求。今天介绍的这款ROG幻X 2025&#xff0c;TA把堪比台式机的性能揉进了1.2kg的轻盈机身中&#xff0c;让…

作者头像 李华