zoukankan      html  css  js  c++  java
  • 力扣No.37 解数独

    编写一个程序,通过填充空格来解决数独问题。

    一个数独的解法需遵循如下规则:

    数字 1-9 在每一行只能出现一次。
    数字 1-9 在每一列只能出现一次。
    数字 1-9 在每一个以粗实线分隔的 3x3 宫内只能出现一次。
    

    空白格用 '.' 表示。

     

    思路:可以用搜索的方法来实现,首先用三个boolean数组记录下该数字在某行、某列、某九宫格里面是否出现过。进入搜索判断该格子是否有数字,如果有就进入到右边的格子,如果没有就用1-9每个数字试一次,如果成功就往下走。如果该位置所有数字都无法填入,就退出返回。当搜索到最右边的格子,就到下一行;如果搜索到表格底部,就返回退出。

    public void solveSudoku(char[][] board) {
            //分别为行、列、小九宫格记录用过的格子
            boolean[][] rowUsed = new boolean[9][9];
            boolean[][] colUsed = new boolean[9][9];
            boolean[][][] boxUsed = new boolean[3][3][9];
            for (int i = 0; i < board.length; i++) {
                for (int j = 0; j < board[0].length; j++) {
                    //格子里的数字为1-9,下标为0-8,所以减1
                    int num = board[i][j] - '1';
                    //如果是数字就设定为该格子有效
                    if (0 <= num && num < 9) {
                        rowUsed[i][num] = true;
                        colUsed[j][num] = true;
                        boxUsed[i / 3][j / 3][num] = true;
                    }
                }
            }
            recusiveSolveSudoku(board, rowUsed, colUsed, boxUsed, 0, 0);
        }
    
        private boolean recusiveSolveSudoku(char[][] board, boolean[][] rowUsed, boolean[][] colUsed, boolean[][][] boxUsed, int row, int col) {
            //如果列走到末尾则重置为零,同时设置到下一行
            if (col == board[0].length) {
                col = 0;
                row++;
                //如果到最后一行说明该解有效
                if (row == board.length) {
                    return true;
                }
            }
            //如果该格子为空
            if (board[row][col] == '.') {
                //从1-9一个个试
                for (int num = 0; num < 9; num++) {
                    //判断该位置用这个数字,行、列、九宫格内有无重复
                    boolean canUsed = !(rowUsed[row][num] || colUsed[col][num] || boxUsed[row / 3][col / 3][num]);
                    if (canUsed) {
                        //设为已用
                        rowUsed[row][num] = true;
                        colUsed[col][num] = true;
                        boxUsed[row / 3][col / 3][num] = true;
                        //索引位置是0-8,加1变为1-9
                        board[row][col] = (char) ('1' + num);
                        //往下递归,如果该位置往下的搜索都能成功,则该位置的数字有效
                        if (recusiveSolveSudoku(board, rowUsed, colUsed, boxUsed, row, col + 1)) {
                            return true;
                        }
                        //返回递归前状态
                        board[row][col] = '.';
                        rowUsed[row][num] = false;
                        colUsed[col][num] = false;
                        boxUsed[row / 3][col / 3][num] = false;
                    }
                }
            }else {
                //该位置有数字了,往右搜索
                return recusiveSolveSudoku(board, rowUsed, colUsed, boxUsed, row, col + 1);
            }
            //如果所有数字都不能满足
            return false;
        }
  • 相关阅读:
    不用写Windows服务实现定时器功能(FluentScheduler )
    (转).NET开发人员必备的可视化调试工具(你值的拥有)
    《C#本质论》读书笔记(14)支持标准查询操作符的集合接口
    关闭 Visual Studio 2013 的 Browser Link 功能
    《C#本质论》读书笔记(12)委托和Lambda表达式
    (2)Underscore.js常用方法
    JS中级
    .NET开发工具之Excel导出公共类
    (1)Underscore.js入门
    datatable绑定comboBox显示数据[C#]
  • 原文地址:https://www.cnblogs.com/RS-Sakura/p/13956258.html
Copyright © 2011-2022 走看看