题解 | #Sudoku#回溯法解数独

Sudoku

http://www.nowcoder.com/practice/78a1a4ebe8a34c93aac006c44f6bf8a1

import java.util.Scanner;

// 注意类名必须为 Main, 不要有任何 package xxx 信息
public class Main {
    public static void main(String[] args) {
        Scanner in = new Scanner(System.in);
        int[][] board = new int[9][9];
        for(int i=0;i<9;i++){
            for(int j=0;j<9;j++){
                board[i][j] = in.nextInt();
            }
        }
        solveSudoku(board);
        //输出二维矩阵
        for(int i=0;i<9;i++){
            for(int j=0;j<8;j++){
                System.out.print(board[i][j] + " ");
            }
            System.out.println(board[i][8]);//换行,每一行的最后一个数字
        }
    }
    public static boolean solveSudoku(int[][] board){
        //「一个for循环遍历棋盘的行,一个for循环遍历棋盘的列,
        // 一行一***定下来之后,递归遍历这个位置放9个数字的可能性!」
        for (int i = 0; i < 9; i++){ // 遍历行
            for (int j = 0; j < 9; j++){ // 遍历列
                if (board[i][j] != 0){ // 跳过原始数字
                    continue;
                }
                for (int k = 1; k <= 9; k++){ // (i, j) 这个位置放k是否合适
                    if (isValidSudoku(i, j, k, board)){
                        board[i][j] = k;//将k放在(i,j)
                        if (solveSudoku(board)){ // 如果找到合适一组立刻返回
                            return true;
                        }
                        board[i][j] = 0;//回溯
                    }
                }
                // 9个数都试完了,都不行,那么就返回false
                return false;
                // 因为如果一行一***定下来了,这里尝试了9个数都不行,说明这个棋盘找不到解决数独问题的解!
                // 那么会直接返回, 「这也就是为什么没有终止条件也不会永远填不满棋盘而无限递归下去!」
            }
        }
        // 遍历完没有返回false,说明找到了合适棋盘位置了
        return true;
    }

    /**
     * 判断棋盘是否合法有如下三个维度:
     *     同行是否重复
     *     同列是否重复
     *     9宫格里是否重复
     */
    public static boolean isValidSudoku(int row, int col, int val, int[][] board){
        // 同行是否重复
        for (int i = 0; i < 9; i++){
            if (board[row][i] == val){
                return false;
            }
        }
        // 同列是否重复
        for (int j = 0; j < 9; j++){
            if (board[j][col] == val){
                return false;
            }
        }
        // 9宫格里是否重复
        int startRow = (row / 3) * 3;
        int startCol = (col / 3) * 3;
        for (int i = startRow; i < startRow + 3; i++){
            for (int j = startCol; j < startCol + 3; j++){
                if (board[i][j] == val){
                    return false;
                }
            }
        }
        return true;
    }
}
全部评论
看见二维就头疼😩😩😩😩
2
送花
回复
分享
发布于 2022-02-09 22:30
问一下,27行 跳过原始数字 和36行回溯是为什么要这么操作呢?
1
送花
回复
分享
发布于 2022-03-08 16:55
秋招专场
校招火热招聘中
官网直投
写的不错
点赞
送花
回复
分享
发布于 2021-12-21 21:51
好棒
点赞
送花
回复
分享
发布于 2022-01-11 23:10
666.可以
点赞
送花
回复
分享
发布于 2022-06-10 19:13
大佬,好牛!!!
点赞
送花
回复
分享
发布于 2022-08-26 14:19 陕西
妙啊
点赞
送花
回复
分享
发布于 2022-10-01 16:05 河南
厉害,我想的是把剩余要填的位置索引记录下来,剩下要填的1-9个数也记录下来 ,再来尝试每种可能是否符合数独,发现直接超时~~。
点赞
送花
回复
分享
发布于 2023-02-17 10:58 山西
太妙了,if(solveSudoku(board)){return true;}board[i][j]=0这两行代码到底是怎么想出来的啊,好牛
点赞
送花
回复
分享
发布于 2023-03-21 00:45 湖南
69行和70行看不懂,求解
点赞
送花
回复
分享
发布于 2023-08-14 12:56 浙江
这样直接填,不会出错吗
点赞
送花
回复
分享
发布于 04-08 12:07 辽宁

相关推荐

点赞 评论 收藏
转发
头像
不愿透露姓名的神秘牛友
05-24 12:16
点赞 评论 收藏
转发
46 11 评论
分享
牛客网
牛客企业服务