leetcode37. 解数独(java)

解数独

  • 解数独
    • 题目描述
    • 回溯算法
    • 代码演示
  • 回溯算法

解数独

难度 困难
leetcode37. 解数独

题目描述

编写一个程序,通过填充空格来解决数独问题。
数独的解法需 遵循如下规则:
1.数字 1-9 在每一行只能出现一次。
2.数字 1-9 在每一列只能出现一次。
3.数字 1-9 在每一个以粗实线分隔的 3x3 宫内只能出现一次。(请参考示例图)
数独部分空格内已填入了数字,空白格用 ‘.’ 表示。

示例1:
leetcode37. 解数独(java)_第1张图片
输入:board = [[“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”]]
输出:[[“5”,“3”,“4”,“6”,“7”,“8”,“9”,“1”,“2”],[“6”,“7”,“2”,“1”,“9”,“5”,“3”,“4”,“8”],[“1”,“9”,“8”,“3”,“4”,“2”,“5”,“6”,“7”],[“8”,“5”,“9”,“7”,“6”,“1”,“4”,“2”,“3”],[“4”,“2”,“6”,“8”,“5”,“3”,“7”,“9”,“1”],[“7”,“1”,“3”,“9”,“2”,“4”,“8”,“5”,“6”],[“9”,“6”,“1”,“5”,“3”,“7”,“2”,“8”,“4”],[“2”,“8”,“7”,“4”,“1”,“9”,“6”,“3”,“5”],[“3”,“4”,“5”,“2”,“8”,“6”,“1”,“7”,“9”]]
解释:输入的数独如上图所示,唯一有效的解决方案如下所示:
leetcode37. 解数独(java)_第2张图片

提示:
board.length == 9
board[i].length == 9
board[i][j] 是一位数字或者 ‘.’
题目数据 保证 输入数独仅有一个解

回溯算法

这题让我们对给定 board 求数独,由于 board 固定是 9 * 9 的大小,我们可以使用回溯算法去做。
对每一个需要填入数字的位置进行填入,如果发现填入某个数会导致数独解不下去,则进行回溯。

由于我们可以填写的数字范围为 [1,9],而数组的下标从 0 开始,因此在存储时,我们使用一个长度为 9 的布尔类型的数组,其中 i个元素的值为 True,当且仅当数字 i+1出现过。例如我们用 line[2][3]=True表示数字 4在第 2 行已经出现过,那么当我们在遍历到第 2 行的空白格时,就不能填入数字 4。

我们首先对整个数独数组进行遍历,当我们遍历到第 i 行第 j 列的位置:

  1. 如果该位置是一个空白格,那么我们将其加入一个用来存储空白格位置的列表中,方便后续的递归操作;

2.如果该位置是一个数字 x,那么我们需要将 line[i][x−1],column[j][x−1] 以及 block[⌊i/3⌋][⌊j/3⌋][x−1]均置为 True。

当我们结束了遍历过程之后,就可以开始递归枚举。当递归到第 iii 行第 jjj 列的位置时,我们枚举填入的数字 x。根据题目的要求,数字 x 不能和当前行、列、九宫格中已经填入的数字相同,因此 line[i][x−1],column[j][x−1]以及 block[⌊i/3⌋][⌊j/3⌋][x−1] 必须均为 False.

当我们填入了数字 x 之后,我们要将上述的三个值都置为 Tru,并且继续对下一个空白格位置进行递归。在回溯到当前递归层时,我们还要将上述的三个值重新置为 False。

代码演示

class Solution {
    boolean[][]col = new boolean[9][9];
    boolean[][]row = new boolean[9][9];
    boolean[][][]ceil = new boolean[3][3][9];

    /**
     * 解数独
     * @param board
     */
    public void solveSudoku(char[][] board) {
       for (int i = 0;i < 9;i++){
           for (int j = 0;j < 9;j++){
               if (board[i][j] != '.'){
                   int num = board[i][j] - '1';
                   row[i][num] = true;
                   col[j][num] = true;
                   ceil[i / 3][j / 3][num] = true;
               }
           }
       }
       dfs(board,0,0);
    }

    public boolean dfs(char[][]board,int x,int y){
        if (y == 9){
            return dfs(board,x + 1,0);
        }
        if (x == 9){
            return true;
        }
        if (board[x][y] != '.'){
            return dfs(board,x,y + 1);
        }
        for (int i = 0;i < 9;i++){
            if (!row[x][i] && !col[y][i] && !ceil[x / 3][y / 3][i]){
                board[x][y] = (char)(i + '1');
                row[x][i] = true;
                col[y][i] = true;
                ceil[x / 3][y / 3][i] = true;
                if (dfs(board,x,y + 1)){
                    break;
                }else {
                    board[x][y] = '.';
                    row[x][i] = false;
                    col[y][i] = false;
                    ceil[x / 3][y / 3][i] = false;
                }

            }
        }
        return board[x][y] != '.';
    }

}

回溯算法

leetcode301. 删除无效的括号

leetcode22. 括号生成

leetcode17. 电话号码的字母组合

你可能感兴趣的:(数据结构,java,算法,java,开发语言,leetcode,数据结构,动态规划,决策树)