LeetCode 第 51 题:n 皇后问题

传送门:51. N皇后

n 皇后问题研究的是如何将 n 个皇后放置在 n×n 的棋盘上,并且使皇后彼此之间不能相互攻击。

img

上图为 8 皇后问题的一种解法。

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

每一种解法包含一个明确的 n 皇后问题的棋子放置方案,该方案中 'Q''.' 分别代表了皇后和空位。

示例:

输入: 4
输出: [
 [".Q..",  // 解法 1
  "...Q",
  "Q...",
  "..Q."],

 ["..Q.",  // 解法 2
  "Q...",
  "...Q",
  ".Q.."]
]
解释: 4 皇后问题存在两个不同的解法。

分析:对于某些计算问题而言,回溯法是一种可以找出所有(或一部分)解的一般性算法,尤其适用于约束满足问题(在解决约束满足问题时,我们逐步构造更多的候选解,并且在确定某一部分候选解不可能补全成正确解之后放弃继续搜索这个部分候选解本身及其可以拓展出的子候选解,转而测试其他的部分候选解)。

回溯法采用试错的思想,它尝试分步的去解决一个问题。在分步解决问题的过程中,当它通过尝试发现现有的分步答案不能得到有效的正确的解答的时候,它将取消上一步甚至是上几步的计算,再通过其它的可能的分步解答再次尝试寻找问题的答案。回溯法通常用最简单的递归方法来实现,在反复重复上述的步骤后可能出现两种情况:

  1. 找到一个可能存在的正确的答案
  2. 在尝试了所有可能的分步方法后宣告该问题没有答案

在最坏的情况下,回溯法会导致一次复杂度为指数时间的计算。

八皇后问题是应用回溯法求解的典型案例。

这篇文章给以 LeetCode 第 51 题为背景,示例了 n 皇后问题的求解。

我们介绍 3 种解法,其核心思想是递归回溯算法。这 3 种解法的不同之处,就在于判断皇后棋子是否可以摆放上,其中用到的方法都很基础,我个人认为都不难理解,并且都应该掌握。

解法 1 :在尝试摆放皇后棋子之前,判断当前考虑的这个位置是否可以放置皇后棋子。

Java 代码:

import java.util.ArrayList;
import java.util.Arrays;
import java.util.LinkedList;
import java.util.List;

// https://leetcode-cn.com/problems/n-queens/description
// 只用一张棋盘摆放的方法完成了 n 皇后问题
public class Solution3 {

    private List<List<String>> res;
    private boolean[][] solution; // true 就表示当前位置放置了皇后,false 就表示没有放置皇后

    public List<List<String>> solveNQueens(int n) {
        res = new ArrayList<>();
        solution = new boolean[n][n];
        nQueens(0, n, solution);
        return res;
    }

    private void nQueens(int row, int n, boolean[][] solution) {
        if (row == n) {
            // 将 solution 处理成为一个棋盘 generateChessboard
            List<String> board = generateChessboard(solution, n);
            res.add(board);
            return;
        }

        // 站在当前这一行(索引为 row),去一列一列检查是否可以放皇后
        for (int i = 0; i < n; i++) {
            if (notInDanger(row, i, n, solution)) {
                solution[row][i] = true;
                nQueens(row + 1, n, solution);
                solution[row][i] = false;// 因为下一列的放置与上一列应该是在同样的环境下进行考虑,棋盘要重置
            }
            // 如果每一列都不能放置,那么这个方法就自动退出了,当前错误的棋盘摆放就不会被保存
        }
    }

    private List<String> generateChessboard(boolean[][] solution, int n) {
        List<String> res = new ArrayList<>();
        StringBuilder stringBuilder;
        for (int i = 0; i < n; i++) {
            stringBuilder = new StringBuilder();
            for (int j = 0; j < n; j++) {
                if (solution[i][j]) {
                    stringBuilder.append("Q");
                } else {
                    stringBuilder.append(".");
                }
            }
            res.add(stringBuilder.toString());
        }
        return res;
    }

    // i,j 表示当前尝试放置皇后棋子的坐标
    private boolean notInDanger(int i, int j, int n, boolean[][] solution) {
        // 设置一些标志,看看当前位置是不是危险的
        // 从之前递归的写法来看,一定是处于不同行的,所以要接着判断,属于不同的列,不同的主对角线和副对角线
        // 判断在 (2,3) 之前的那些行,在 3 这一列上是否有皇后,即 (0,3)、 (1,3)
        for (int r = i; r >= 0; r--) {
            if (solution[r][j]) {
                return false;
            }
        }
        // 判断在 (2,3) 之前的那些行,在它的主对角线上是否有皇后(向右上方走)
        for (int r = i, c = j; r >= 0 && c < n; r--, c++) {
            if (solution[r][c]) {
                return false;
            }
        }
        // 判断在 (2,3) 之前的那些行,在它的副对角线上是否有皇后(向左上方走)
        for (int r = i, c = j; r >= 0 && c >= 0; r--, c--) {
            if (solution[r][c]) {
                return false;
            }
        }
        // 全部判断完以后,才表示安全
        return true;
    }

    public static void main(String[] args) {
        Solution3 solution = new Solution3();
        List<List<String>> solveNQueens = solution.solveNQueens(8);
        int solveSize = solveNQueens.size();
        System.out.println("一共有 " + solveSize + " 种解。");

        for (int i = 0; i < solveNQueens.size(); i++) {
            System.out.println("第 " + (i + 1) + " 种解:");
            for (String line : solveNQueens.get(i)) {
                System.out.println(line);
            }
        }
    }
}
  • 我个人觉得这种解法使用了典型的回溯方法的套路。我们看递归方法 private void nQueens(int row, int n, boolean[][] solution)
    我们观察这个方法是如何调用自己的:在一个循环中递归调用自己,而仅仅只是参数不同,在调用自己前设置状态,在调用自己后取消状态。是不是非常像回溯法解决排列问题。

解法 2:使用三个一维数组记录了是否可以摆放皇后棋子。

下面这种解法,在主对角线和副对角线上是否可以摆放元素使用了并不太难的技巧,使得求解更加简洁。

Java 代码:

import java.util.ArrayList;
import java.util.Arrays;
import java.util.LinkedList;
import java.util.List;

// https://leetcode-cn.com/problems/n-queens/description
// 刘宇波老师介绍的 n 皇后问题的解法
public class Solution2 {

    private List<List<String>> res;
    private boolean[] col;
    private boolean[] dia1;
    private boolean[] dia2;

    public List<List<String>> solveNQueens(int n) {
        res = new ArrayList<>();
        col = new boolean[n];
        // 可以用归纳法计算出对角线数组元素的个数为 2*n-1
        // 这里要动手用纸笔计算一下,并不难
        dia1 = new boolean[2 * n - 1];
        dia2 = new boolean[2 * n - 1];
        // 表示一行数据,第 0 个数字表示"第 0 行的皇后应该摆放在哪个索引位置"
        LinkedList<Integer> row = new LinkedList<>();
        putQueen(n, 0, row);
        return res;
    }

    // 在一个 n 皇后问题中,尝试在第 index 行如何摆放一个皇后
    // 我们将它设计成一个递归调用的函数,所以要想清楚两个步骤
    private void putQueen(int n, int index, LinkedList<Integer> row) {
        if (index == n) {
            res.add(generateBoard(n, row));
        }
        for (int i = 0; i < n; i++) { // i 表示列数,index 表示行数
            if (!col[i] && !dia1[index + i] && !dia2[index - i + n - 1]) {
                // 设置状态
                row.addLast(i);
                col[i] = true;
                dia1[index + i] = true;
                dia2[index - i + n - 1] = true;
                putQueen(n, index + 1, row);
                // 重置状态(与前面设置状态的步骤是一一对应的)
                col[i] = false;
                dia1[index + i] = false;
                dia2[index - i + n - 1] = false;
                row.removeLast();
            }
        }
    }

    private List<String> generateBoard(int n, LinkedList<Integer> row) {
        List<String> board = new ArrayList<>();
        for (int i = 0; i < n; i++) {
            char[] charArray = new char[n];
            Arrays.fill(charArray, '.');
            charArray[row.get(i)] = 'Q';
            board.add(new String(charArray));
        }
        return board;
    }

    public static void main(String[] args) {
        Solution2 solution = new Solution2();
        List<List<String>> solveNQueens = solution.solveNQueens(8);
        int solveSize = solveNQueens.size();
        System.out.println("一共有 " + solveSize + " 种解。");

        for (int i = 0; i < solveNQueens.size(); i++) {
            System.out.println("第 " + (i + 1) + " 种解:");
            for (String line : solveNQueens.get(i)) {
                System.out.println(line);
            }
        }
    }
}

解法 3 :使用一个二维数组记录了每添加一个皇后,棋盘中被已经存在的皇后可以攻击到的范围。

下面的这种写法可能看起来篇幅比较长,但是其中用到的使用方向数组进行状态设置的技巧还是可以参考的。

另外,还要注意,下面的这个方法涉及到二维数组的复制,Java 中并不支持直接对二维数组进行深复制,因此最简单的做法,就是将二维数组中的元素一个一个进行复制。

Java 代码:

import java.util.ArrayList;
import java.util.List;

// https://leetcode-cn.com/problems/n-queens/description/
// 这一版使用了递归回溯的思想完成
// 使用 marked 数组把皇后四面八方的攻击范围都给标记上了
public class Solution {

    //  x-1,y-1   x-1,y  x-1,y+1
    //  x,y-1     x,y    x,y+1
    //  x+1,y-1   x+1,y  x+1,y+1
    private static int[] dx = new int[]{-1, -1, -1, 0, 0, 1, 1, 1};
    private static int[] dy = new int[]{-1, 0, 1, -1, 1, -1, 0, 1};

    private int n;

    // 当前坐标 x,y 放上了皇后以后,它的 8 个方向都应该被标记不能放置其它皇后
    // 标记 marked 数组,给皇后的四面八方都标记上
    private void put_down_the_queen(int x, int y, boolean[][] marked) {
        marked[x][y] = true;
        // 从 1 开始, 从 0 开始也可以,只不过重复了 marked[x][y] = true;
        for (int i = 1; i < n; i++) {
            for (int j = 0; j < 8; j++) {
                // 不要忘记乘以 i
                int new_x = x + i * dx[j];
                int new_y = y + i * dy[j];
                // 只要都在棋盘内
                if (new_x >= 0 && new_x < n && new_y >= 0 && new_y < n) {
                    marked[new_x][new_y] = true;
                }
            }
        }
    }

    public List<List<String>> solveNQueens(int n) {
        this.n = n;
        List<List<String>> res = new ArrayList<>();
        // 默认值是 false ,表示当前没有放置皇后
        boolean[][] marked = new boolean[n][n];

        // n 皇后问题的一个解
        char[][] solution = new char[n][n];
        for (int i = 0; i < n; i++) {
            for (int j = 0; j < n; j++) {
                solution[i][j] = '.';
            }
        }
        generate(0, n, marked, solution, res);
        return res;
    }

    // 在第 k 行(从 0 开始计算)考虑,这一行的 n 个元素,搜索皇后应该摆放在哪里
    // 皇后摆放的位置应该在 marked 数组中标记为 false 的地方,此时已经摆放皇后的情况位于 solution 数组中
    // 所有已经得到的 n 皇后问题的解位于 res 中
    private void generate(int k, int n, boolean[][] marked, char[][] solution, List<List<String>> res) {
        if (k == n) {
            res.add(transform2str(solution));
            return;
        }
        for (int i = 0; i < n; i++) {
            if (!marked[k][i]) { // 如果没有标记过,表示当前位置是安全的,可以放置皇后
                // 注意:对于二维数组的复制,不能简单实用 clone() 或者 System.arraycopy() 或者 Arrays.copyOf() 方法
                // 应该对它们在一维数组上使用
                boolean[][] temp_mark = copyNewArr(marked);
                solution[k][i] = 'Q';
                put_down_the_queen(k, i, marked);
                generate(k + 1, n, marked, solution, res);
                marked = temp_mark;
                solution[k][i] = '.';
            }
        }
    }

    // 复制一个 marked 数组
    private boolean[][] copyNewArr(boolean[][] marked) {
        boolean[][] res = new boolean[n][n];
        for (int i = 0; i < n; i++) {

            System.arraycopy(marked[i],0,res[i],0,n);
        }
        return res;
    }

    private List<String> transform2str(char[][] solution) {
        List<String> res = new ArrayList<>();
        for (int i = 0; i < n; i++) {
            res.add(new String(solution[i]));
        }
        return res;
    }

    public static void main(String[] args) {
        Solution solution = new Solution();
        List<List<String>> solveNQueens = solution.solveNQueens(8);
        int solveSize = solveNQueens.size();
        System.out.println("一共有 " + solveSize + " 种解。");

        for (int i = 0; i < solveNQueens.size(); i++) {
            System.out.println("第 " + (i + 1) + " 种解:");
            for (String line : solveNQueens.get(i)) {
                System.out.println(line);
            }
        }
    }
}

(本节完)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,732评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,496评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,264评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,807评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,806评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,675评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,029评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,683评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,704评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,666评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,773评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,413评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,016评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,204评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,083评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,503评论 2 343

推荐阅读更多精彩内容