n皇后问题是将n个皇后放置在n*n的棋盘上,皇后彼此之间不能相互攻击(任意两个皇后不能位于同一行,同一列,同一斜线)。

给定一个整数n,返回所有不同的n皇后问题的解决方案。

每个解决方案包含一个明确的n皇后放置布局,其中“Q”和“.”分别表示一个女王和一个空位置。

在线评测地址:LintCode 领扣

样例1:

输入:1
输出:[["Q"]]

样例2:

输入:4
输出:
[// Solution 1[".Q..","...Q","Q...","..Q."],// Solution 2["..Q.","Q...","...Q",".Q.."]
]

算法:dfs(回溯法)

题目分析

这个问题要求把n个皇后放在一个nXn的棋盘上,使得任何两个皇后都不能相互攻击,即它们不能同行,不能同列,也不能位于同一条对角线上。对于n=1,问题的解很简单,而且很容易看出对于n=2和n=3来说,这个问题是无解的。所以我们考虑4皇后问题,并用回溯法对它求解。

算法思路

  • 因为每个皇后都必须分别占据一行,我们需要做的不过是棋盘上的每个皇后分配一列。
  • 下面我们用4皇后的求解过程来讲解算法思路:

从空棋盘开始,然后把皇后1 放到它所在行的第-一个可能位置上,也就是第一-行第一列。对于皇后2,在经过第-列和第二列的失败尝试之后,我们把它放在第一个可能的位置,就是格子(2, 3),位于第二行第三列的格子。这被证明是一个死胡同,因为皇后3将没有位置可放。所以,该算法进行回溯,把皇后2放在下一个可能位置(2,4)上。这样皇后3就可以放在(3, 2),这被证明是另一个死胡同。该算法然后就回溯到底,把皇后1移到(1,2)。 接着皇后2到(2,4), 皇后3到(3,1), 而皇后4到(4, 3), 这就是该问题的一个解。

  • 整个过程实际上就是一个状态树的遍历过程
  • 下图为状态树

代码思路

  • 按行摆放,在确定一个皇后应该摆的列时,需要检查当前列是否合法,如果合法,则将皇后放置在当前位置,并进行递归,回溯。每行都摆满皇后时,则产生了一种解法,将所有解法收集并返回。
  • 合法性判断方法:当前将要摆放皇后的位置和其他已摆放皇后的位置不能在同一列,且不能在同一条斜线上。这里判断是否在同一条斜线上可以通过两个皇后的位置横坐标之差和纵坐标之差的绝对值是否相等来判断。

复杂度分析

  • 空间复杂度:O(N!)
  • 时间复杂度:O(N!)
  • 放置第一个皇后有 N 种可能,放置两个皇后不超过N(N-2)种可能,放置三个皇后不超过N(N - 2)(N - 4)种可能 ,以此类推。
class Solution {/*** Get all distinct N-Queen solutions* @param n: The number of queens* @return: All distinct solutions* For example, A string '...Q' shows a queen on forth position*/List<List<String>> solveNQueens(int n) {// result用于存储答案List<List<String>> results = new ArrayList<>();if (n <= 0) {return results;}search(results, new ArrayList<Integer>(), n);return results;}// search函数为搜索函数,n表示已经放置了n个皇后,cols 表示每个皇后所在的列private void search(List<List<String>> results, List<Integer> cols, int n) {// 若已经放置了n个皇后表示出现了一种解法,绘制后加入答案resultif (cols.size() == n) {results.add(Draw(cols));return;}// 枚举当前皇后放置的列,若不合法则跳过for (int colIndex = 0; colIndex < n; colIndex++) {if (!isValid(cols, colIndex)) {continue;}// 若合法则递归枚举下一行的皇后cols.add(colIndex);search(results, cols, n);cols.remove(cols.size() - 1);}}// isValid函数为合法性判断函数private boolean isValid(List<Integer> cols, int col) {int row = cols.size();for (int rowIndex = 0; rowIndex < cols.size(); rowIndex++) {//若有其他皇后在同一列或同一斜线上则不合法if (cols.get(rowIndex) == col) {return false;}if (row + col == rowIndex + cols.get(rowIndex)) {return false;}if (row - col == rowIndex - cols.get(rowIndex)) {return false;}}return true;}// Draw函数为将 cols 数组转换为答案的绘制函数private List<String> Draw(List<Integer> cols) {List<String> result = new ArrayList<>();for (int i = 0; i < cols.size(); i++) {StringBuilder sb = new StringBuilder();for (int j = 0; j < cols.size(); j++) {sb.append(j == cols.get(i) ? 'Q' : '.');}result.add(sb.toString());}return result;}
}

更多题解参考:九章算法

八皇后时间复杂度_九章算法 | N皇后问题相关推荐

  1. n皇后问题c语言_九章算法 | N皇后问题

    n皇后问题是将n个皇后放置在n*n的棋盘上,皇后彼此之间不能相互攻击(任意两个皇后不能位于同一行,同一列,同一斜线). 给定一个整数n,返回所有不同的n皇后问题的解决方案. 每个解决方案包含一个明确的 ...

  2. 动态规划法求最大字段和时间复杂度_九章算法 | 动态规划:最长上升子序列

    给定一个整数序列,找到最长上升子序列(LIS),返回LIS的长度. 在线评测地址:LintCode 领扣 说明 最长上升子序列的定义: 最长上升子序列问题是在一个无序的给定序列中找到一个尽可能长的由低 ...

  3. 最长回文子串动态规划_九章算法 | 微软面试题:最长回文子串

    给出一个字符串(假设长度最长为1000),求出它的最长回文子串,你可以假定只有一个满足条件的最长回文串. 在线评测地址:LintCode 领扣 样例 1: 输入:"abcdzdcab&quo ...

  4. 编号是i的结点所在的层次号是_九章算法 | 微软面试题:二叉树的锯齿形层次遍历...

    给出一棵二叉树,返回其节点值的锯齿形层次遍历(先从左往右,下一层再从右往左,层与层之间交替进行) 在线评测地址:LintCode 领扣 样例 1: 输入:{1,2,3} 输出:[[1],[3,2]] ...

  5. 多个数字数组_九章算法 | 谷歌面试题:多个数组的交集

    给出多个数组,求它们的交集.输出他们交集的大小. 在线评测地址:LintCode 领扣 样例 1: 输入: [[1,2,3],[3,4,5],[3,9,10]]输出: 1解释:只有3出现在三个数组中. ...

  6. arrays合并两个数组_九章算法 | 字节跳动面试题:合并k个排序数组

    将 k 个有序数组合并为一个大的有序数组. 在线评测地址:LintCode 领扣 样例 1: Input: [[1, 3, 5, 7],[2, 4, 6],[0, 8, 9, 10, 11]] Out ...

  7. 九章算法【总结】Java 搞定链表-面试常考题目精选

    面试大总结之链表 CS3K.com 一.OverView: 链表是面试中常考的,本文参考了其它一些文章,加上小编的自己总结,基本每个算法都测试并优化过. 算法大全(1)单链表 中还有一些链表题目,将来 ...

  8. 分治习题--九章算法培训课第三章笔记

    1.Maximum Depth of Binary Tree 这是道简单的分治习题了 分: 左子树最大深度 右子树最大深度 治: 最大深度等于max(左子树,右子树)+1 public class S ...

  9. 九章算法 | 苏州微软面试题:程序检查

    描述 有一种编程语言,只有以下五种命令,每种命令最多有两个参数,请检查给定的程序是否 可能 无限循环. 这些命令分别是: label <string>:声明一个标签,参数是一个字符串,且每 ...

最新文章

  1. tcpdump-根据IP查看程序与服务都用了哪些端口
  2. 获取并编译linux源码,android获取源代码、编译、命令
  3. 什么是.hpp文件?
  4. 清零 css,css样式清零及常用类
  5. 流浪地球开机动画包zip_【文娱热点】流浪地球2定档2023大年初一;迪士尼计划裁员32000人...
  6. 工作的时候用到spring返回xml view查到此文章亲测可用
  7. 【数据结构笔记20】图的定义,图的表示:邻接矩阵与邻接表
  8. php访问mysql 封装
  9. 离散数学-集合运算基本法则
  10. C#中常用字符串操作
  11. 桌面养花-DesktopPlant v2.3.12原创汉化版
  12. B-002 电容基础知识
  13. 初级第二旬04—六字准提观试题
  14. 以前的我们——那年大一
  15. 达摩院量子计算机叫什么,刚刚,阿里巴巴达摩院宣布研制出全球最强量子电路模拟器...
  16. OSG KML文件解析
  17. L1、L2正则化总结
  18. Newton迭代法求解Toeplitz矩阵逆的程序
  19. Android:logo
  20. Ubuntu设置apt源

热门文章

  1. 百度云盘免下载百度云盘全速下载方法
  2. 第 二 十 八 天 :监 控 软 件 之 cacti
  3. 精心整理的十个必须要知道CSS+DIV技巧
  4. POJ 1185 炮兵阵地(动态规划+状态压缩)
  5. 桌面云的四大协议解析
  6. 浅谈C#实现Web代理服务器的几大步骤
  7. 【4】CCNA理论第三天
  8. lua 去除小数点有效数字后面的0_【物联网学习番外篇】Lua脚本编程扫盲
  9. laravel连接oracle6,Laravel 使用 Oracle 数据库
  10. 定时任务_定时任务Quartz入门