题目描述
-
编写一个程序,通过已填充的空格来解决数独问题。
-
一个数独的解法需遵循如下规则:
数字 1-9 在每一行只能出现一次。
数字 1-9 在每一列只能出现一次。
数字 1-9 在每一个以粗实线分隔的 3x3 宫内只能出现一次。 -
PS:空白格用 ‘.’ 表示。
代码
public static void main(String[] args) {
char[][] board = new char[][]{
{'5', '3', '.', '.', '7', '.', '.', '.', '.'},
{'6', '.', '.', '1', '9', '5', '.', '.', '.'},
{'.', '9', '8', '.', '.', '.', '.', '6', '.'},
{'8', '.', '.', '.', '6', '.', '.', '.', '3'},
{'4', '.', '.', '8', '.', '3', '.', '.', '1'},
{'7', '.', '.', '.', '2', '.', '.', '.', '6'},
{'.', '6', '.', '.', '.', '.', '2', '8', '.'},
{'.', '.', '.', '4', '1', '9', '.', '.', '5'},
{'.', '.', '.', '.', '8', '.', '.', '7', '9'}
};
sovle(board);
}
private static void sovle(char[][] board) {
//从00 位置开始深搜
dfs(board,0,0);
}
private static void dfs(char[][] board, int x, int y) {
if (x==9) {
//使用退出而不是return 所以打印在此处
for (int i = 0; i < 9; i++) {
for (int j = 0; j < 9; j++) {
System.out.print(board[i][j]+" ");
}
System.out.println();
}
System.exit(0);
}
//当前位置为空
if(board[x][y]=='.') {
for (int i = 1; i < 10; i++) {
//确认 当前位置是否可以放
if(check(board,x,y,i)){
//一定要用‘0’+i 转化为字符存入
board[x][y]=(char) ('0'+i);
//搜索下一个位置 ps:当且仅当y=8时,即y是横排最后一个元素,x才+1,y才回溯为0;
dfs(board, x+(y+1)/9, (y+1)%9);
}
}
//代表至此位置1-9都不可填;
board[x][y]='.';
}else {
//此空不为空,搜索下一个位置
dfs(board, x+(y+1)/9, (y+1)%9);
}
}
private static boolean check(char[][] board, int x, int y, int i) {
//横排
for (int j = 0; j <9; j++) {
if(board[x][j]=='0'+i)
return false;
}
//列
for (int j2 = 0; j2 < 9; j2++) {
if(board[j2][y]=='0'+i)
return false;
}
//所在九宫格
for (int j = x/3*3; j <x/3*3+3; j++) {
for (int j2 = y/3*3; j2 < y/3*3+3; j2++) {
if(board[j][j2]=='0'+i)
return false;
}
}
return true;
}