问题描述
843. n-皇后问题
n−皇后问题是指将 n 个皇后放在 n×n 的国际象棋棋盘上,使得皇后不能相互攻击到,即任意两个皇后都不能处于同一行、同一列或同一斜线上。
现在给定整数 n,请你输出所有的满足条件的棋子摆法。
输入格式
共一行,包含整数 n。
输出格式
每个解决方案占 n 行,每行输出一个长度为 n 的字符串,用来表示完整的棋盘状态。
其中 . 表示某一个位置的方格状态为空,Q 表示某一个位置的方格上摆着皇后。
每个方案输出完成后,输出一个空行。
注意:行末不能有多余空格。
输出方案的顺序任意,只要不重复且没有遗漏即可。
数据范围
1≤n≤9
输入样例:
4
输出样例:
.Q.....QQ.....Q...Q.Q......Q.Q..
思路
与组合问题类似,找到一种能遍历所有情况的方法。
方法一按格子遍历,最大深度为n*n
方法二按行遍历,最大深度为n
可见不同的遍历顺序效果也不尽相同。
注意:在dfs过程中存储中间状态和结果。
方法一
import java.util.*;public class Main {static int n;static char[][] g;static int[] row, col, dg, udg;public static void main(String[] args) {Scanner sc = new Scanner(System.in);n = sc.nextInt();g = new char[n][n];col = new int[n];row = new int[n];dg = new int[2 * n];udg = new int[2 * n];for (int i = 0; i < n; i++) {Arrays.fill(g[i], '.');}dfs(0, 0, 0);}static void dfs(int x, int y, int sum) {if (y == n) {y = 0;x++;}if (x == n) {if (sum == n) {for (int i = 0; i < n; i++)System.out.println(g[i]);System.out.println();}return;}dfs(x, y + 1, sum);if (row[x] == 0 && col[y] == 0 && dg[y + x] == 0 && udg[y - x + n] == 0) {row[x] = col[y] = dg[y + x] = udg[y - x + n] = 1;g[x][y] = 'Q';dfs(x, y+1, sum + 1);g[x][y] = '.';row[x] = col[y] = dg[y + x] = udg[y - x + n] = 0;}}}
方法二
import java.util.*;public class Main {static int n;static char[][] g;static int[] col, dg, udg;public static void main(String[] swe) {Scanner sc = new Scanner(System.in);n = sc.nextInt();g = new char[n][n];col = new int[n];dg = new int[2 * n];udg = new int[2 * n];for (int i = 0; i < n; i++) {for (int j = 0; j < n; j++) {g[i][j] = '.';}}dfs(0);}static void dfs(int u) {if (u == n) {for (int i = 0; i < n; i++) {System.out.println(g[i]);}System.out.println();return;}for (int i = 0; i < n; i++) {if (col[i] == 0 && dg[i + u] == 0 && udg[i - u + n] == 0) {g[u][i] = 'Q';col[i] = 1;dg[i + u] = 1;udg[i - u + n] = 1;dfs(u + 1);g[u][i] = '.';col[i] = 0;dg[i + u] = 0;udg[i - u + n] = 0;}}}}
