皇后之战:揭秘N皇后问题的多维解法与智慧【python 力扣52题】

ops/2024/10/18 9:22:30/

作者介绍:10年大厂数据\经营分析经验,现任大厂数据部门负责人。
会一些的技术:数据分析、算法、SQL、大数据相关、python
欢迎加入社区:码上找工作
作者专栏每日更新:
LeetCode解锁1000题: 打怪升级之旅
python数据分析可视化:企业实战案例
备注说明:方便大家阅读,统一使用python,带必要注释,公众号 数据分析螺丝钉 一起打怪升级

题目描述

给定一个整数 n,返回 n×n 棋盘上的 N 皇后问题的不同解决方案的数量。

每一种解法包含一个明确的 n×n 棋盘上的皇后放置方式,其中 'Q' 代表一个皇后,而 '.' 代表空位。每个皇后都必须满足不能与其他皇后在同一行、同一列或同一对角线上。

输入格式
  • n:一个整数,表示棋盘的大小。
输出格式
  • 返回所有独特的 n 皇后问题的解决方案数量。

示例

示例 1

在这里插入图片描述

输入: n = 4
输出: 2
示例 2
输入: n = 1
输出: 1

注意
LeetCode题目51 "N皇后"与题目52 "N皇后II"虽然都是基于经典的N皇后问题,但两者的主要区别在于输出的需求不同:

N皇后 (题目 51)

  • 目标:找出所有可能的N皇后解决方案的具体棋盘布局。
  • 输出:返回所有解决方案,每个解决方案都详细展示了皇后的具体摆放位置。每个解决方案是一个包含n个字符串的列表,每个字符串长度为n,表示棋盘的一行,其中’Q’表示皇后,'.'表示空位。
  • 解法关注点:除了求解算法的有效性外,还需要关注如何构造并展示完整的棋盘布局。

N皇后 II (题目 52)

  • 目标:仅计算N皇后问题的不同解决方案数量。
  • 输出:返回解决方案的数量,而不是具体的棋盘布局。
  • 解法关注点:主要关注于优化算法的效率以快速计算出解决方案的总数,而不需要构造棋盘的具体布局。
    解法对比
    尽管两个问题在解法上可能使用类似的技术(如回溯法),51题需要更多的空间和逻辑来存储和展示所有可能的棋盘配置,而52题则更注重于计数优化,可能会使用更加精简的数据结构(如位运算)来加速计数过程。

方法一:回溯算法

解题步骤
  1. 初始化变量:创建一个用于存储当前行皇后位置的列表。
  2. 定义回溯函数:递归定义函数以尝试每一行的每个位置。
  3. 合法性检查:检查当前位置放置皇后是否合法,即检查列和两个方向的对角线。
  4. 递归与计数:递归地放置下一个皇后,如果完成一种有效布局则增加计数。
完整的规范代码
python">def totalNQueens(n):"""计算 n 皇后问题的解决方案总数:param n: int, 棋盘大小:return: int, 解决方案总数"""def backtrack(row, diagonals, anti_diagonals, cols):if row == n:nonlocal countcount += 1returnfor col in range(n):curr_diagonal = row - colcurr_anti_diagonal = row + colif (col in cols or curr_diagonal in diagonals or curr_anti_diagonal in anti_diagonals):continuecols.add(col)diagonals.add(curr_diagonal)anti_diagonals.add(curr_anti_diagonal)backtrack(row + 1, diagonals, anti_diagonals, cols)cols.remove(col)diagonals.remove(curr_diagonal)anti_diagonals.remove(curr_anti_diagonal)count = 0backtrack(0, set(), set(), set())return count# 示例调用
print(totalNQueens(4))  # 输出: 2
算法分析
  • 时间复杂度:(O(n!)),尽管有剪枝,每层递归的选择数接近 n
  • 空间复杂度:(O(n)),递归栈深度加用于存储状态的空间。

方法二:位运算优化的回溯算法

解题步骤
  1. 位运算优化:使用位运算代替集合操作,提高效率。
  2. 定义位运算回溯函数:使用位掩码表示列和对角线的占用状态,通过位运算快速检查和修改状态。
  3. 递归与计数:递归放置皇后,完成布局时增加解决方案计数。
完整的规范代码
python">def totalNQueens(n):"""计算 n 皇后问题的解决方案总数,使用位运算进行优化:param n: int, 棋盘大小:return: int, 解决方案总数"""def solve(row, hills, next_row, dales):if row == n:nonlocal countcount += 1returnfree_columns = columns & ~(hills | next_row | dales)while free_columns:curr_column = -free_columns & free_columnssolve(row + 1, (hills | curr_column) << 1, next_row | curr_column, (dales | curr_column) >> 1)free_columns &= free_columns - 1columns = (1 << n) - 1count = 0solve(0, 0, 0, 0)return count# 示例调用
print(totalNQueens(4))  # 输出: 2
算法分析
  • 时间复杂度:(O(n!)),位运算显著提高了效率,但最坏情况下仍需尝试所有可能。
  • 空间复杂度:(O(n)),递归深度决定了空间复杂度,虽然使用位运算减少了空间占用。

方法三:迭代回溯

解题步骤
  1. 使用栈模拟递归:使用栈来模拟递归过程,避免函数调用的开销。
  2. 迭代处理:在迭代中管理棋盘状态和递归变量,以模拟递归调用栈的行为。
完整的规范代码
python">def totalNQueens(n):"""使用迭代回溯解决 n 皇后问题:param n: int, 棋盘大小:return: int, 解决方案总数"""stack = [(0, 0, 0, 0)]  # (row, hills, next_row, dales)count = 0while stack:row, hills, next_row, dales = stack.pop()if row == n:count += 1continuefree_columns = ((1 << n) - 1) & ~(hills | next_row | dales)while free_columns:curr_column = -free_columns & free_columnsstack.append((row + 1, (hills | curr_column) << 1, next_row | curr_column, (dales | curr_column) >> 1))free_columns &= free_columns - 1return count# 示例调用
print(totalNQueens(4))  # 输出: 2
算法分析
  • 时间复杂度:(O(n!)),迭代的方式减少了递归调用开销,但仍然需要尝试所有可能的放置方式。
  • 空间复杂度:(O(n)),虽然使用栈来模拟递归,但空间复杂度与递归方法相当。

不同算法的优劣势对比

特征方法一: 回溯算法方法二: 位运算优化回溯方法三: 迭代回溯
时间复杂度(O(n!))(O(n!))(O(n!))
空间复杂度(O(n))(O(n))(O(n))
优势- 易于理解和实现- 空间效率高- 避免递归调用开销
劣势- 空间消耗较大- 理解和实现较为复杂- 状态维护较为复杂

应用示例

科学研究
N皇后问题常用于算法研究和教学,特别是在探讨组合数学、算法优化、复杂度分析等领域。此问题的不同解决策略可用于教授递归、回溯及其优化。

算法竞赛
算法竞赛中,N皇后问题是经典问题,经常出现在各类比

赛和面试中,作为测试程序员解决复杂问题能力的一种方式。

通过以上方法和示例,可以深入理解和掌握N皇后问题的多种解决方案及其应用场景。这些技术不仅限于此问题,还可广泛应用于其他需要递归和回溯解决的问题中。


http://www.ppmy.cn/ops/5370.html

相关文章

Ollama教程——使用langchain:ollama与langchain的强强联合

相关文章: Ollama教程——入门&#xff1a;开启本地大型语言模型开发之旅 Ollama教程——模型&#xff1a;如何将模型高效导入到ollama框架 Ollama教程——兼容OpenAI API&#xff1a;高效利用兼容OpenAI的API进行AI项目开发 Ollama教程——使用langchain&#xff1a;ollama与…

快速删除node_modules依赖包的命令rimraf

1、安装rimraf npm install -g rimraf 2、使用命令删除node_modules rimraf node_modules *** window系统&#xff0c;使用命令很快就删除node_modules ***

第二十六章: mybatis plus 如何使用`LambdaQueryWrapper` 和 `QueryWrapper`

第二十六章&#xff1a; mybatis plus 如何使用LambdaQueryWrapper 和 QueryWrapper 目标 掌握 LambdaQueryWrapper 和 QueryWrapper的用法掌握 List对象转map对象掌握 List对象获取某字段的集合 LambdaQueryWrapper 和 QueryWrapper 是 MyBatis-Plus 中提供的查询条件构造器…

笔记本电脑上的聊天机器人: 在英特尔 Meteor Lake 上运行 Phi-2

对应于其强大的能力&#xff0c;大语言模型 (LLM) 需要强大的算力支撑&#xff0c;而个人计算机上很难满足这一需求。因此&#xff0c;我们别无选择&#xff0c;只能将它们部署至由本地或云端托管的性能强大的定制 AI 服务器上。 为何需要将 LLM 推理本地化 如果我们可以在典配…

【干货精品分享】Elasticsearch 6.7 Should 子语句的失效

在ES 使用多条件 查询&#xff0c;并且是多个条件只需要满足部分条件的时候&#xff0c;我们通常会使用到ES的should查询 GET /trademark_query_index/_search {"query":{"bool" : {"must":[{"match" : {"origin": {"…

Redis 核心知识点常考面试题(持续更新中)

Redis 核心知识点常考面试题&#xff08;持续更新中&#xff09; Redis单线程IO多路复用原理Redis缓存穿透、缓存雪崩、缓存击穿问题Redis与数据库双写不一致问题基于Redis实现分布式锁的的应用场景Redis持久化方式Redis内存淘汰机制Redis删除策略Redis主从复制、哨兵、集群Red…

TCP/IP_第八章_静态路由_实验案例一

实验案例一&#xff1a;配置静态路由实现全网互通 1、实验环境 如图8.10所示&#xff0c;三台路由器R1&#xff0e;R2&#xff0c;R3两两互连&#xff0c;每台路由器上都配置了Loopback地址模拟网络环境。 2、需求描述 需要在三台路由器上配置静态路由&#xff0c;以实现各网…

CSS中:root伪类的说明和使用

定义和用法 :root选择器用匹配文档的根元素。在HTML中根元素始终是HTML元素&#xff0c;所以也可以把:root理解为html根元素选择器&#xff0c;但是比html根元素的优先级高&#xff0c;:root伪类选择器常常被用于定义全局的CSS变量或者设置全局的CSS样式。CSS :root 选择器 | …