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

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 的数字或 ' . ' 。

 bool isValidSudoku(vector<vector<char>>& board) {
vector<char> row, column, subbox;
for (int i = ; i < ; ++i) {
row.clear();
column.clear();
row = board[i];
if (!isValid(row))
return false;
for (int j = ; j < ; ++j) {
column.push_back(board[j][i]);
}
if (!isValid(column))
return false;
}
for (int i = ; i <= ; i = i + ) {
for (int j = ; j <= ; j = j + ) {
subbox.clear();
subbox.push_back(board[i][j]); subbox.push_back(board[i][j+]); subbox.push_back(board[i][j+]);
subbox.push_back(board[i+][j]); subbox.push_back(board[i+][j+]); subbox.push_back(board[i+][j+]);
subbox.push_back(board[i+][j]); subbox.push_back(board[i+][j+]); subbox.push_back(board[i+][j+]);
if (!isValid(subbox))
return false;
}
}
return true;
} bool isValid(vector<char> &t) {
sort(t.begin(), t.end());
for (int i = ; i < ; ++i) {
if (((t[i] < '' || t[i] > '') && t[i] != '.') || (i > && t[i] == t[i - ] && t[i] != '.'))
return false;
}
return true;
}

答案巧妙的做法如下

 bool isValidSudoku(vector<vector<char>>& board) {
bool row[][] = {false}, col[][] = {false}, box[][] = {false};
for (int i = ; i < ; i++) {
for (int j = ; j < ; j++) {
if (board[i][j] != '.') {
int num = board[i][j] - '' - , k = i / * + j / ;
if (row[i][num] || col[j][num] || box[k][num]) return false;
row[i][num] = col[j][num] = box[k][num] = true;
}
}
}
return true;
}

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实现也很巧妙

 public boolean isValidSudoku(char[][] board) {
for(int i = 0; i < 9; i++) {
Set<Character> rows = new HashSet<>();
Set<Character> cols = new HashSet<>();
Set<Character> cubes = new HashSet<>();
for (int j = 0; j < 9; j++) {
if (board[i][j] != '.' && !rows.add(board[i][j])) return false;
if (board[j][i] != '.' && !cols.add(board[j][i])) return false;
int colStart = 3 * (i % 3), rowStart = 3 * (i / 3);
int colOffset = j % 3, rowOffset = j / 3; // 偏移
int row = rowStart + rowOffset, col = colStart + colOffset;
if (board[row][col] != '.' && !cubes.add(board[row][col]) ) return false;
}
}
return true;
}

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

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

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

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

通过 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】数独的更多相关文章

  1. [LeetCode] Sudoku Solver 求解数独

    Write a program to solve a Sudoku puzzle by filling the empty cells. Empty cells are indicated by th ...

  2. [LeetCode] Valid Sudoku 验证数独

    Determine if a Sudoku is valid, according to: Sudoku Puzzles - The Rules. The Sudoku board could be ...

  3. LeetCode:36. Valid Sudoku,数独是否有效

    LeetCode:36. Valid Sudoku,数独是否有效 : 题目: LeetCode:36. Valid Sudoku 描述: Determine if a Sudoku is valid, ...

  4. LeetCode 37 Sudoku Solver(求解数独)

    题目链接: https://leetcode.com/problems/sudoku-solver/?tab=Description   Problem : 解决数独问题,给出一个二维数组,将这个数独 ...

  5. LeetCode 36 Valid Sudoku(合法的数独)

    题目链接: https://leetcode.com/problems/valid-sudoku/?tab=Description   给出一个二维数组,数组大小为数独的大小,即9*9  其中,未填入 ...

  6. [LeetCode] “全排列”问题系列(一) - 用交换元素法生成全排列及其应用,例题: Permutations I 和 II, N-Queens I 和 II,数独问题

    一.开篇 Permutation,排列问题.这篇博文以几道LeetCode的题目和引用剑指offer上的一道例题入手,小谈一下这种类型题目的解法. 二.上手 最典型的permutation题目是这样的 ...

  7. “全排列”问题系列(一)[LeetCode] - 用交换元素法生成全排列及其应用,例题: Permutations I 和 II, N-Queens I 和 II,数独问题

    转:http://www.cnblogs.com/felixfang/p/3705754.html 一.开篇 Permutation,排列问题.这篇博文以几道LeetCode的题目和引用剑指offer ...

  8. Leetcode之回溯法专题-37. 解数独(Sudoku Solver)

    Leetcode之回溯法专题-37. 解数独(Sudoku Solver) 编写一个程序,通过已填充的空格来解决数独问题. 一个数独的解法需遵循如下规则: 数字 1-9 在每一行只能出现一次.数字 1 ...

  9. [LeetCode] 37. Sudoku Solver 求解数独

    Write a program to solve a Sudoku puzzle by filling the empty cells. A sudoku solution must satisfy  ...

  10. [LeetCode] 36. Valid Sudoku 验证数独

    Determine if a 9x9 Sudoku board is valid. Only the filled cells need to be validated according to th ...

随机推荐

  1. ABAP-FI常用BAPI

    总帐会计:  (比较简单全部测试通过,关帐时使用) Line item of document for ledger with summary table GL F: BAPI_GLX_GETDOCI ...

  2. transient关键字详解

    作用 1,一旦变量被transient修饰,变量将不再是对象持久化的一部分,该变量内容在序列化后无法获得访问. 2,transient关键字只能修饰变量,而不能修饰方法和类.注意,本地变量是不能被tr ...

  3. string截取、替换、查找子串函数,find_first_of 用法

    1. 截取子串 s.substr(pos, n) 截取s中从pos开始(包括0)的n个字符的子串,并返回 s.substr(pos) 截取s中从从pos开始(包括0)到末尾的所有字符的子串,并返回 2 ...

  4. 用 Anaconda 完美解决 Python2 和 python3 共存问题

    Python3 被越来越多的开发者所接受,同时让人尴尬的是很多遗留的老系统依旧运行在 Python2 的环境中,因此有时你不得不同时在两个版本中进行开发,调试. 如何在系统中同时共存 Python2 ...

  5. 前端如何应对笔试算法题?(用node编程)

    用nodeJs写算法题 咱们前端使用算法的地方不多,但是为了校招笔试,不得不针对算法题去练习呀! 好不容易下定决心 攻克算法题.发现js并不能像c语言一样自建输入输出流.只能回去学习c语言了吗?其实不 ...

  6. 理解 Redis(9) - Publish Subscribe 消息订阅

    在窗口1开通一个名为 redis 的通道: 127.0.0.1:6379> SUBSCRIBE redis Reading messages... (press Ctrl-C to quit) ...

  7. scikit_learn逻辑回归类库

    来自:刘建平 1.概述 在scikit-learn中,与逻辑回归有关的主要有3个类.LogisticRegression, LogisticRegressionCV 和 logistic_regres ...

  8. JavaScript 局部刷新

    JavaScript局部刷新具体代码展示如下 1.  #tabList代表需要刷新的元素的对象 2.  第二个#tabList 如果后面有第三个元素,那么后面需要加>*符号,如果不加,容易造成C ...

  9. hive表的存储路径查找以及表的大小

    1.在hive中知道一个表的存储路径可以通过hive命令   desc formatted table_name 显示表的详细信息; 2.然后找到该表的存储路径 "Location:    ...

  10. DRF框架固定配置

    1.认证Authentication 可以在配置文件中配置全局默认的认证方案 REST_FRAMEWORK = { 'DEFAULT_AUTHENTICATION_CLASSES': ( 'rest_ ...