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

2024-04-21 03:36

本文主要是介绍皇后之战:揭秘N皇后问题的多维解法与智慧【python 力扣52题】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

作者介绍: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. 递归与计数:递归地放置下一个皇后,如果完成一种有效布局则增加计数。
完整的规范代码
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. 递归与计数:递归放置皇后,完成布局时增加解决方案计数。
完整的规范代码
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. 迭代处理:在迭代中管理棋盘状态和递归变量,以模拟递归调用栈的行为。
完整的规范代码
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皇后问题的多种解决方案及其应用场景。这些技术不仅限于此问题,还可广泛应用于其他需要递归和回溯解决的问题中。

这篇关于皇后之战:揭秘N皇后问题的多维解法与智慧【python 力扣52题】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/922067

相关文章

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

Python MySQL如何通过Binlog获取变更记录恢复数据

《PythonMySQL如何通过Binlog获取变更记录恢复数据》本文介绍了如何使用Python和pymysqlreplication库通过MySQL的二进制日志(Binlog)获取数据库的变更记录... 目录python mysql通过Binlog获取变更记录恢复数据1.安装pymysqlreplicat

利用Python编写一个简单的聊天机器人

《利用Python编写一个简单的聊天机器人》这篇文章主要为大家详细介绍了如何利用Python编写一个简单的聊天机器人,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 使用 python 编写一个简单的聊天机器人可以从最基础的逻辑开始,然后逐步加入更复杂的功能。这里我们将先实现一个简单的

基于Python开发电脑定时关机工具

《基于Python开发电脑定时关机工具》这篇文章主要为大家详细介绍了如何基于Python开发一个电脑定时关机工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 简介2. 运行效果3. 相关源码1. 简介这个程序就像一个“忠实的管家”,帮你按时关掉电脑,而且全程不需要你多做

Python实现高效地读写大型文件

《Python实现高效地读写大型文件》Python如何读写的是大型文件,有没有什么方法来提高效率呢,这篇文章就来和大家聊聊如何在Python中高效地读写大型文件,需要的可以了解下... 目录一、逐行读取大型文件二、分块读取大型文件三、使用 mmap 模块进行内存映射文件操作(适用于大文件)四、使用 pand

python实现pdf转word和excel的示例代码

《python实现pdf转word和excel的示例代码》本文主要介绍了python实现pdf转word和excel的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价... 目录一、引言二、python编程1,PDF转Word2,PDF转Excel三、前端页面效果展示总结一

Python xmltodict实现简化XML数据处理

《Pythonxmltodict实现简化XML数据处理》Python社区为提供了xmltodict库,它专为简化XML与Python数据结构的转换而设计,本文主要来为大家介绍一下如何使用xmltod... 目录一、引言二、XMLtodict介绍设计理念适用场景三、功能参数与属性1、parse函数2、unpa

Python中使用defaultdict和Counter的方法

《Python中使用defaultdict和Counter的方法》本文深入探讨了Python中的两个强大工具——defaultdict和Counter,并详细介绍了它们的工作原理、应用场景以及在实际编... 目录引言defaultdict的深入应用什么是defaultdictdefaultdict的工作原理