资源限制
时间限制:1.0s 内存限制:512.0MB
问题描述
给定一个n*n的棋盘,棋盘中有一些位置不能放皇后。现在要向棋盘中放入n个黑皇后和n个白皇后,使任意的两个黑皇后都不在同一行、同一列或同一条对角线上,任意的两个白皇后都不在同一行、同一列或同一条对角线上。问总共有多少种放法?n小于等于8。
输入格式
输入的第一行为一个整数n,表示棋盘的大小。
接下来n行,每行n个0或1的整数,如果一个整数为1,表示对应的位置可以放皇后,如果一个整数为0,表示对应的位置不可以放皇后。
输出格式
输出一个整数,表示总共有多少种放法。
样例输入
4
1 1 1 1
1 1 1 1
1 1 1 1
1 1 1 1
样例输出
2
样例输入
4
1 0 1 1
1 1 1 1
1 1 1 1
1 1 1 1
样例输出
0
AC代码
package main;
import java.awt.Point;
import java.util.*;
class Puzzle {
int[][] map;
int n;
boolean[][] isVisited;
int sumOfWays;
boolean[] isRowHaveBlackKing, isColumnHaveBlackKing, isRowHaveWhiteKing, isColumnHaveWhiteKing;
boolean[] isLeftDiagonalHaveBlackKing, isRightDiagonalHaveBlackKing, isRightDiagonalHaveWhiteKing, isLeftDiagonalHaveWhiteKing;
public Puzzle(int n) {
this.n = n;
sumOfWays = 0;
map = new int[n][n];
isVisited = new boolean[n][n];
isRowHaveBlackKing = new boolean[n];
isRowHaveWhiteKing = new boolean[n];
isColumnHaveBlackKing = new boolean[n];
isColumnHaveWhiteKing = new boolean[n];
isRightDiagonalHaveBlackKing = new boolean[3 * n];
isRightDiagonalHaveWhiteKing = new boolean[3 * n];
isLeftDiagonalHaveBlackKing = new boolean[3 * n];
isLeftDiagonalHaveWhiteKing = new boolean[3 * n];
}
public void cinPuzzle(Scanner cin) {
for (int i = 0; i < n; i++) {
for (int j = 0; j < n; j++) {
map[i][j] = cin.nextInt();
}
}
}
public void putWhiteKing(int rowCount) {
if (rowCount >= n) {
sumOfWays++;
return;
}
for (int i = 0; i < n; i++) {
if (map[rowCount][i] == 0) {
continue;
}
if (isVisited[rowCount][i]) {
continue;
}
if (isColumnHaveWhiteKing[i] || isLeftDiagonalHaveWhiteKing[rowCount - i + n] || isRightDiagonalHaveWhiteKing[rowCount + i]) {
continue;
}
isColumnHaveWhiteKing[i] = true;
isLeftDiagonalHaveWhiteKing[rowCount - i + n] = true;
isRightDiagonalHaveWhiteKing[rowCount + i] = true;
isVisited[rowCount][i] = true;
putWhiteKing(rowCount + 1);
isColumnHaveWhiteKing[i] = false;
isLeftDiagonalHaveWhiteKing[rowCount - i + n] = false;
isRightDiagonalHaveWhiteKing[rowCount + i] = false;
isVisited[rowCount][i] = false;
}
}
public void putBlackKing(int rowCount) {
if (rowCount >= n) {
putWhiteKing(0);
return ;
}
for (int i = 0; i < n; i++) {
if (map[rowCount][i] == 0) {
continue;
}
if (isColumnHaveBlackKing[i] || isLeftDiagonalHaveBlackKing[rowCount - i + n] || isRightDiagonalHaveBlackKing[rowCount + i]) {
continue;
}
isColumnHaveBlackKing[i] = true;
isLeftDiagonalHaveBlackKing[rowCount - i + n] = true;
isRightDiagonalHaveBlackKing[rowCount + i] = true;
isVisited[rowCount][i] = true;
putBlackKing(rowCount + 1);
isColumnHaveBlackKing[i] = false;
isRightDiagonalHaveBlackKing[rowCount + i] = false;
isLeftDiagonalHaveBlackKing[rowCount - i + n] = false;
isVisited[rowCount][i] = false;
}
}
}
public class Main {
public static void main(String[] args) {
Scanner cin = new Scanner(System.in);
Puzzle puzzle = new Puzzle(cin.nextInt());
puzzle.cinPuzzle(cin);
puzzle.putBlackKing(0);
System.out.println(puzzle.sumOfWays);
cin.close();
}
}