题目
难度困难445收藏分享切换为英文关注反馈
编写一个程序,通过已填充的空格来解决数独问题。
一个数独的解法需遵循如下规则:
- 数字
1-9
在每一行只能出现一次。 - 数字
1-9
在每一列只能出现一次。 - 数字
1-9
在每一个以粗实线分隔的3x3
宫内只能出现一次。
空白格用 '.'
表示。
一个数独。
答案被标成红色。
来源:力扣(LeetCode)37.解数独
链接:https://leetcode-cn.com/problems/sudoku-solver/
Note:
- 给定的数独序列只包含数字
1-9
和字符'.'
。 - 你可以假设给定的数独只有唯一解。
- 给定数独永远是
9x9
形式的。
分析
见注释
解答
class Solution {
// 判断 board[i][j] 是否可以填入 number
boolean canFillNumber(char[][] board, int row, int col, char number) {
for (int k = 0; k < 9; k++) {
// 判断所在行 的每一列 是否已经存在该数字
if (board[row][k] == number) return false;
// 判断所在列 的每一行 是否存在重复
if (board[k][col] == number) return false;
// 判断 3 x 3 方框是否存在重复
if (board[(row / 3) * 3 + k / 3][(col / 3) * 3 + k % 3] == number)
return false;
}
return true;
}
boolean backtrack(char[][] board, int i, int j) {
if (j == 9) {
// 每次穷举到最后一列的话就换到下一行重新开始。
return backtrack(board, i + 1, 0);
}
if (i == 9) {
// 最后一行时, 表示找到一个可行解,触发 base case
return true;
}
if (board[i][j] != '.') {
// 如果有预设填写的数字,不用我们穷举, 直接下一列
return backtrack(board, i, j + 1);
}
for (char number = '1'; number <= '9'; number++) {
// 从1到9, 看看可不可以填, 如果不可以填写,则下一个
if (!canFillNumber(board, i, j, number))
continue;
// 填数字试一下
board[i][j] = number;
// 如果找到一个可行解,立即结束
if (backtrack(board, i, j + 1)) {
return true;
}
// 撤销选择
board[i][j] = '.';
}
// 穷举完 1~9,依然没有找到可行解,此路不通
return false;
}
public void solveSudoku(char[][] board) {
backtrack(board, 0, 0);
System.out.println(board);
}
}