LeetCode 图解 | 36.有效的数独

  • 2020 年 2 月 20 日
  • 筆記

以下文章来源于算法无遗策 ,作者我脱下短袖

今天分享一个LeetCode题,题号是36,标题是:有效的数独,题目标签是散列表,散列表也称哈希表。此题解题思路用到了少量的空间换取时间的方法,降低时间上的消耗

题目描述

判断一个 9×9 的数独是否有效。只需要根据以下规则,验证已经填入的数字是否有效即可。

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

数独

上图是一个部分填充的有效的数独。

数独部分空格内已填入了数字,空白格用 '.' 表示。

示例 1:

输入:  [    ["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"]  ]    输出: true

示例 2:

输入:  [    ["8","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"]  ]    输出: false    解释: 除了第一行的第一个数字从 5 改为 8 以外,空格内其他数字均与 示例1 相同。       但由于位于左上角的 3x3 宫内有两个 8 存在, 因此这个数独是无效的。

说明:

一个有效的数独(部分已被填充)不一定是可解的。    只需要根据以上规则,验证已经填入的数字是否有效即可。    给定数独序列只包含数字 1-9 和字符 '.' 。    给定数独永远是 9x9 形式的。

解题

此题没有要求数独是可解的,只要求满足以下规则,验证已经填入的数字是否有效即可:

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

行的下标设为i,列的下标设为j,宫格的下标设为k,默认为0,如下图:

行、列和宫格

随着下标i和下标j的移动,i和j可以直接从下标中获取数字,但k如何获取对应的数字呢?看上面图,k随着i变化和k随着j变化都有规律的,不多说,直接给公式:k = (int)(i / 3) * 3 + (int)(j / 3)。

根据规则,某数字的三个下标都只能出现一次,例如8:[0,0,0],往后这个数组里就不能再出现0了,有0出现就不符合有效数独的规则了;再例如3:[0,1,0],i下标和k下标不能再出现0了,j下标不能再出现1了。

但怎么判断某数字的三个下标是否是只出现了一次呢?

题目标签只有散列表,那正合我意,我就是要用散列表去解决此题。而且数组里的值最小是0,最大值是8,数组的长度都固定为3,可以用少量的空间换取时间的方法,如下图8:[0,0,0]的表示:

空间换时间

这样就减少了两个数组比较的烦恼,通过空间换取时间的方法,就减少了不必要的比较计算。因为行i、列j和宫格k的长度都是9,将二维数组摊开作为一维数组,下标i、下标j+9和下标k+18分别控制一维数组的下标,存放的值都是布尔类型,默认为false。

保存某数字的时候,一维数组的下标i、下标j+9和下标k+18的值都变为true。保存某数字之前,需要判断三个下标的值是否存在true,如果不存在,则将三个下标对应的值都变为true;如果存在,说明某下标已经出现一次了,再出现一次则意味着这个数独已经无效,直接返回false。如下图数字8的下标k已经出现一次了。

失效的数独

动画:使用散列表
Code:使用散列表
public boolean isValidSudoku(char[][] board) {      // 创建散列表      Map<Integer, boolean[]> map = new HashMap<>();      for (int i = 0; i < 9; i++) {          for (int j = 0; j < 9; j++) {              if (board[i][j] != '.') {                  // 字符的ASCII码 十进制                  int index = board[i][j];                  // 创建宫格标记                  int k = i / 3 * 3 + j / 3;                  // 空间换取时间                  if (!map.containsKey(index)) {                      map.put(index, new boolean[27]); // 27个空间默认放false                  }                  // 获取散列表的值                  boolean[] booleans = map.get(index);                  if (booleans[i] == true || booleans[j + 9] == true || booleans[k + 18] == true) {                      return false;                  } else {                      booleans[i] = true;                      booleans[j + 9] = true;                      booleans[k + 18] = true;                  }              }          }      }      return true;  }    public static void main(String[] args) {      char[][] 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'}      };      boolean validSudoku = new Solution().isValidSudoku(board);      System.out.println(validSudoku);  }
时间复杂度

时间复杂度是O(n),但实际上比O(n)要快。