news 2026/1/2 6:54:07

八皇后问题:回溯算法精解

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
八皇后问题:回溯算法精解

八皇后问题包含了回溯算法的核心思想 ——试探 - 回溯 - 剪枝

  • 同一行:每行仅放 1 个皇后(按行遍历可天然避免行冲突);
  • 同一列:需标记已占用的列,避免重复使用;
  • 对角线:棋盘有两条对角线方向(左上→右下、右上→左下),需分别标记占用状态。

1. 回溯法的核心逻辑

回溯法本质是一种 “暴力搜索的优化方案”,通过深度优先搜索(DFS)试探每一种可能,当发现当前路径无法达到目标时,立即 “回溯” 到上一步,尝试其他路径,避免无效搜索。

对于 8 皇后问题:

  • 按行递归:每行仅放 1 个皇后,递归层数 = 棋盘行数(8 层);
  • 按列试探:每一行遍历 8 列,尝试在无冲突的列放置皇后;
  • 冲突检测:通过标记数组快速判断列和对角线是否占用(O (1) 时间复杂度);
  • 回溯恢复:递归返回后,撤销当前列和对角线的占用标记,恢复棋盘状态,为下一列试探做准备。

2. 关键优化:冲突检测的标记数组设计

(1)列标记数组colUsed
  • 作用:标记某一列是否已放置皇后;
  • 大小:8(棋盘列数),索引对应列号(0~7);
  • 状态:colUsed[col] = true表示第 col 列已占用。
(2)对角线标记数组

棋盘有两条对角线,需分别设计标记数组:

  • 左上→右下对角线(记为 diag1):同一对角线上的皇后满足row - col = 常数

    • 取值范围:row-col 的结果为 -7~7(8 行 8 列),加 7 转为非负索引(0~14),故数组大小为 15;
    • 索引计算:diag1 = row - col + 7
  • 右上→左下对角线(记为 diag2):同一对角线上的皇后满足row + col = 常数

    • 取值范围:row+col 的结果为 0~14,天然非负,数组大小为 15;
    • 索引计算:diag2 = row + col

通过这三个标记数组,可在 O (1) 时间内完成冲突检测,避免了暴力搜索中 “遍历已放皇后判断冲突” 的 O (n) 复杂度,大幅提升效率。

C++代码实现

#include <iostream>
#include <vector>
using namespace std;

int count = 0; // 统计解的数量
vector<bool> colUsed; // 标记列是否被占用(大小8)
vector<bool> diag1Used; // 标记左上→右下对角线(大小15,2*8-1=15)
vector<bool> diag2Used; // 标记右上→左下对角线(大小15)
vector<vector<char> > chessboard; // 存储棋盘状态:'Q'=皇后,'.'=空位

// 回溯函数:按行处理,用标记数组快速判断冲突
void backtracking(int row) {
// 终止条件:所有8行放完,找到一个合法解
if (row == 8) {
count++; // 解数+1
// 打印当前合法解的棋盘
cout << "===== 第 " << count << " 个合法解 =====" << endl;
for (int i = 0; i < 8; ++i) { // 遍历每一行
for (int j = 0; j < 8; ++j) { // 遍历每一列
cout << chessboard[i][j] << " "; // 打印当前格(加空格更美观)
}
cout << endl; // 一行结束换行
}
cout << "======================" << endl << endl; // 分隔不同解
return;
}

// 遍历当前行的所有列
for (int col = 0; col < 8; ++col) {
// 计算对角线索引(避免负数)
int diag1 = row - col + 7; // 范围:0~14(原row-col范围-7~7,+7转正)
int diag2 = row + col; // 范围:0~14(天然非负)

// 冲突检测:O(1)快速判断(无冲突则继续)
if (!colUsed[col] && !diag1Used[diag1] && !diag2Used[diag2]) {
// 标记当前位置为皇后
chessboard[row][col] = 'Q';
// 标记当前列和对角线为占用
colUsed[col] = true;
diag1Used[diag1] = true;
diag2Used[diag2] = true;

backtracking(row + 1); // 递归处理下一行

// 回溯:撤销标记,恢复状态(为尝试当前行下一列做准备)
colUsed[col] = false;
diag1Used[diag1] = false;
diag2Used[diag2] = false;
chessboard[row][col] = '.'; // 恢复当前位置为空位
}
}
}
/*
...(逐层递归,直到row=8)
→ row=8 → 打印解 → return
← 回到backtracking(7) → 撤销row=7的状态 → 尝试下一列
← 回到backtracking(6) → 撤销row=6的状态 → 尝试下一列
← 回到backtracking(0) → 撤销row=0, col=0的状态 → 尝试col=1
*/

int main() {
// 初始化标记数组(默认false,代表未占用)
colUsed.resize(8, false);
diag1Used.resize(15, false);
diag2Used.resize(15, false);
// 初始化棋盘:8行8列,所有位置默认设为'.'(空位)
chessboard.resize(8, vector<char>(8, '.'));

backtracking(0); // 从第0行(第一行)开始回溯

cout << "剪枝优化回溯法解数:" << count << endl; // 输出总解数(92)
return 0;
}

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

14、Linux文件系统管理与设备挂载全解析

Linux文件系统管理与设备挂载全解析 1. 磁盘挂载与卸载基础 当使用新磁盘时,需要显式地挂载它。可以使用 umount 命令来卸载磁盘,例如: # umount /dev/fd0 # umount /mnt/floppy对于 umount 或 mount 操作,可以指定挂载的目录或设备,如 /dev/fd0 。卸载后,就…

作者头像 李华
网站建设 2025/12/29 1:03:34

10、Kubernetes入门与有状态工作负载管理

Kubernetes入门与有状态工作负载管理 1. ConfigMap的使用 ConfigMap用于存储非敏感配置数据,方便在Kubernetes中管理和使用。以下是关于ConfigMap的详细介绍及使用方法。 1.1 更新配置 创建ConfigMap后,可以使用 kubectl edit configmap <configmap_name> 命令来…

作者头像 李华
网站建设 2025/12/18 4:54:51

京东秒杀助手:从抢购新手到购物达人的实用指南

京东秒杀助手&#xff1a;从抢购新手到购物达人的实用指南 【免费下载链接】jd-assistant 京东抢购助手&#xff1a;包含登录&#xff0c;查询商品库存/价格&#xff0c;添加/清空购物车&#xff0c;抢购商品(下单)&#xff0c;查询订单等功能 项目地址: https://gitcode.com…

作者头像 李华
网站建设 2025/12/18 4:53:33

10、Linux 家用/办公软件入门指南

Linux 家用/办公软件入门指南 1. 办公生产力软件 在办公软件方面,Linux 系统有多种选择。 1.1 办公套件 LibreOffice :LibreOffice Writer 看起来与 Word 相似,但并不完全相同。它是一套完整的办公软件,涵盖了文字处理、电子表格、演示文稿等多种功能,是比较常用的办…

作者头像 李华
网站建设 2025/12/18 4:53:18

19、Linux系统软件依赖管理与更新升级全解析

Linux系统软件依赖管理与更新升级全解析 1. 软件依赖的概念与查看 软件依赖指的是一个程序运行所需的文件和库(被多个程序共享和使用的模块化软件)。包管理器的部分职责就是处理这些依赖,它会检查一个包运行所需的文件,确保系统中存在这些文件,如果不存在则进行安装,还…

作者头像 李华
网站建设 2025/12/18 4:52:51

如何快速对比PDF文档差异?diff-pdf工具完整使用指南

如何快速对比PDF文档差异&#xff1f;diff-pdf工具完整使用指南 【免费下载链接】diff-pdf A simple tool for visually comparing two PDF files 项目地址: https://gitcode.com/gh_mirrors/di/diff-pdf PDF文档对比是日常办公和学习中经常遇到的需求&#xff0c;无论是…

作者头像 李华