zoukankan      html  css  js  c++  java
  • 【LeetCode】数独

    判断一个数独是否合法,未填的空格用字符 ' . ' 表示。该数独有解并不是必要的。

    e.g. 如图合法数独,输入

    ["53..7....","6..195...",".98....6.","8...6...3","4..8.3..1","7...2...6",".6....28.","...419..5","....8..79"]

    返回 true。

    我依然使用死办法解决,而且进行了输入合法性判断,即必须为 1 ~ 9 的数字或 ' . ' 。

     1 bool isValidSudoku(vector<vector<char>>& board) {
     2     vector<char> row, column, subbox;
     3     for (int i = 0; i < 9; ++i) {
     4         row.clear();
     5         column.clear();
     6         row = board[i];
     7         if (!isValid(row))
     8             return false;
     9         for (int j = 0; j < 9; ++j) {
    10             column.push_back(board[j][i]);
    11         }
    12         if (!isValid(column))
    13             return false;
    14     }
    15     for (int i = 0; i <= 6; i = i + 3) {
    16         for (int j = 0; j <= 6; j = j + 3) {
    17             subbox.clear();
    18             subbox.push_back(board[i][j]);    subbox.push_back(board[i][j+1]);    subbox.push_back(board[i][j+2]);
    19             subbox.push_back(board[i+1][j]);  subbox.push_back(board[i+1][j+1]);  subbox.push_back(board[i+1][j+2]);
    20             subbox.push_back(board[i+2][j]);  subbox.push_back(board[i+2][j+1]);  subbox.push_back(board[i+2][j+2]);
    21             if (!isValid(subbox))
    22                 return false;
    23         }
    24     }
    25     return true;
    26 }
    27     
    28     bool isValid(vector<char> &t) {
    29     sort(t.begin(), t.end());
    30     for (int i = 0; i < 9; ++i) {
    31         if (((t[i] < '1' || t[i] > '9') && t[i] != '.') || (i > 0 && t[i] == t[i - 1] && t[i] != '.'))
    32             return false;
    33     }
    34     return true;
    35 }

        答案巧妙的做法如下

     1 bool isValidSudoku(vector<vector<char>>& board) {
     2     bool row[9][9] = {false}, col[9][9] = {false}, box[9][9] = {false};
     3     for (int i = 0; i < 9; i++) {
     4         for (int j = 0; j < 9; j++) {
     5             if (board[i][j] != '.') {
     6                 int num = board[i][j] - '0' - 1, k = i / 3 * 3 + j / 3;
     7                 if (row[i][num] || col[j][num] || box[k][num]) return false;
     8                 row[i][num] = col[j][num] = box[k][num] = true;
     9             }
    10         }
    11     }
    12     return true;
    13 }

    num = board[i][j] - '0' - 1 的思路就是用一个新 bool 型数组(初始化全为 false)判断原数组是否有重复元素

    k = i / 3 * 3 + j / 3 将原来 9 * 9 的方格映射到 3 * 3 的方格中!

    0 | 1 | 2
    3 | 4 | 5
    6 | 7 | 8

    例如 i = 5,j = 6 (第 5 行 第 6 列)时,k = 5 / 3 * 3 + 6 / 3 = 1 * 3 + 2 = 5,在 box[5][] 这个数组里进行判断。

    这种方法更常规的用法见下。

        这个Java实现也很巧妙

     1 public boolean isValidSudoku(char[][] board) {
     2     for(int i = 0; i < 9; i++) {
     3         Set<Character> rows = new HashSet<>();
     4         Set<Character> cols = new HashSet<>();
     5         Set<Character> cubes = new HashSet<>();            
     6         for (int j = 0; j < 9; j++) {
     7             if (board[i][j] != '.' && !rows.add(board[i][j])) return false;
     8             if (board[j][i] != '.' && !cols.add(board[j][i])) return false;
     9             int colStart = 3 * (i % 3), rowStart = 3 * (i / 3);
    10             int colOffset = j % 3, rowOffset = j / 3;   // 偏移
    11             int row = rowStart + rowOffset, col = colStart + colOffset;
    12             if (board[row][col] != '.' && !cubes.add(board[row][col]) ) return false;
    13         }
    14     }
    15     return true;
    16 }

    HashSet 的 add(E e) 方法用于将指定元素添加到这个 HashSet,若此 Set 已经包含该元素,则直接返回 false。

    %/ 操作符对于矩阵遍历问题很有帮助。

    使用 %水平遍历,即计算列坐标偏移。因为 j 每增加 1,j % 3 也增加 1 然后重置

    使用  /  作竖直遍历,即计算行坐标偏移。因为 j 每增加 3,j / 3 才能增加 1

    通过 0 ~ 8 的 j 即可遍历一个 9 * 9 矩阵的一个 3 * 3 子块。如何继续遍历下一个子块呢?就需要用外层循环 0 ~ 8 的 i 实现。

    依然使用 % 水平遍历到下一个子块,colStart = 3 * (i % 3) ,× 3 是因为下一个子块在 3 列之后,第一个子块的起始是 (0, 0),第二个子块的起始是 (0, 3) 而不是 (0, 1)。

    e.g.

    i = 2 时,j 从 0 ~ 8,

    rowStart = 3 * (2 / 3) = 0                                      colStart = 3 * (2 % 3) = 6

    rowOffset = j / 3 = 0,0,0, 1,1,1, 2,2,2                   colOffset = j % 3 = 0,1,2, 0,1,2, 0,1,2

    对应了 board 矩阵中的

    (0+0, 6+0) (0+0, 6+1) (0+0, 6+2)
    (0+1, 6+0) (0+1, 6+1) (0+1, 6+2)
    (0+2, 6+0) (0+2, 6+1) (0+2, 6+2)

    (0,6) (0,7) (0,8)
    (1,6) (1,7) (1,8)
    (2,6) (2,7) (2,8)

    这个 Sub-Box。

  • 相关阅读:
    LeetCode 905 按奇偶排序数组
    LeetCode 46 全排列
    Django 2随便使用笔记-Day01
    Python函数化编程整理
    Oracle解锁表笔记
    springboot(1)使用SpringBoot基础HTTP接口GET|POST|DELETE|PUT请求
    什么是Restful API
    C# 生成条形码BarCode 128
    ADB shell 的一般操作
    遇到“未能从程序集XXXX...加载类型XXX”的问题
  • 原文地址:https://www.cnblogs.com/wayne793377164/p/7199544.html
Copyright © 2011-2022 走看看