题目:Write a program to solve a Sudoku puzzle by filling the empty cells.

  Empty cells are indicated by the character '.'.

  You may assume that there will be only one unique solution.

下面是一个数独的题目:

其解:

数独不是很了解,没做过。不过知道规则。就是在这个9x9的格纸中间添1到9的数字。使每一行不能重复,每一列也不能重复,然后上面那个粗线框起来的3x3的格纸中的数字也不能重复。

不知道这样的游戏有啥意义。。。

Anyway,这个和那个什么皇后的题很类似,一般用回溯。不过递归显得比较清晰。

添格子么。。。那不是一个一个添。顺序呢?我肯定是从左到右从上到下这么添。那代码我也这样写。对于每个格子,假设前面的所有格子已经添好了,并且构成了一个合理的棋盘。对当前的格子,如果没有填充,在查看该格子所在的行,列还有所在的3x3子格子后,我尝试每个剩余的可用数字,对每个数字,填充当前格子,然后问题就递归地变成了下面一个格子的填充问题。

如下图所示,我们要填充第三个格子。

我们首先查看Cell(0,2)所在的列2,标记出现的数字,也就是8;然后再查看其所在的行0,标记出现的数字5,3和7,最后查看它所在的3x3的格子,有5,3,6,9,和8。这样,3,5,6,7,8,9都被标记了,剩下的就是0,2,4三种可能。

我们不妨选一种,比如4,然后继续递归Cell(0,3)。如果这种选择不合理,在最后没有生成合理的结果,那在递归返回到Cell(0,2)的时候,我们在选择下一个可能的值就是了,例如2.但是这里有个要注意的是,如果当前所有的可能的值尝试都失败,一定要重新恢复当前Cell的值。再返回false。代码如下:

 /**
* 该函数假设从board[0][0]到
* board[row][col]之前的所有
* 格子都已经添满并且构成一个合理的board
* @param board 棋盘
* @param row 当前需要填充的cell的row
* @param col 当前需要填充的cell的col
* @return
*/
private static boolean solve(char[][]board, int row, int col){
int nextCol = (col + 1) % SIZE; //下一个需要填充的格子的列号
int nextRow = col == SIZE-1 ? row + 1 : row; //下一个需要填充的格子的行号
boolean[] set = new boolean[SIZE]; //记录当前格子可用的数字
if(board[row][col] == '.') {
scanColRowGrid(board, row, col, set);
for(int i = 0; i < SIZE; i++){
if(!set[i]){
board[row][col] = (char)('1' + i);
if(nextRow == SIZE) return true;
if(solve(board, nextRow, nextCol)) return true;
//else try next available number in this cell...
}
}
board[row][col] = '.'; //一定要恢复这个值
return false;
}else {
if(nextRow == SIZE) return true;
return solve(board, nextRow, nextCol);
}
}

其中的scanColRowGrid就是决定当前格可添的数字。

 /**
* 决定board[row][col]可填写的数字
* @param board
* @param row
* @param col
* @param set 当函数返回时,set[i]为false表明数字i+1可以填写到棋盘中。
* @return
*/
private static boolean[] scanColRowGrid(char[][] board, int row, int col, boolean[] set){
for(int i = 0; i < SIZE; i++){
if(board[row][i] >= '1' && board[row][i] <= '9') {
set[board[row][i] - '1'] = true;
}
}
for(int i = 0; i < SIZE; i++){
if(board[i][col] >= '1' && board[i][col] <= '9') {
set[board[i][col] - '1'] = true;
}
}
int grid = getGrid(row, col);
int st_row = 3 * (grid / 3);
int st_col = 3 * (grid % 3);
for(int i = 0; i < 3; i++){
for(int j = 0; j < 3; j++){
char c = board[st_row + i][st_col + j];
if(c >= '1' && c <= '9') {
set[c - '1'] = true;
}
}
}
return set;
} /**
* 通过行号和列号查询该格子所在的3x3格
* @param row
* @param col
* @return 0~8的数字,代表从左到右从上到下的3x3格标号
*/
private static int getGrid(int row, int col){
return 3 * (row / 3) + col / 3;
}

注意我们对3x3格的标号方式,是从左到右从上到下,标号为0~8,方便我们转换。

这道题虽然不难,但是比较经典,值得总结。

LeetCode 笔记系列十 Suduko的更多相关文章

  1. LeetCode 笔记系列 18 Maximal Rectangle [学以致用]

    题目: Given a 2D binary matrix filled with 0's and 1's, find the largest rectangle containing all ones ...

  2. LeetCode 笔记系列16.3 Minimum Window Substring [从O(N*M), O(NlogM)到O(N),人生就是一场不停的战斗]

    题目:Given a string S and a string T, find the minimum window in S which will contain all the characte ...

  3. Python基础笔记系列十四:python无缝调用c程序

    本系列教程供个人学习笔记使用,如果您要浏览可能需要其它编程语言基础(如C语言),why?因为我写得烂啊,只有我自己看得懂!! python语言可以对c程序代码进行调用,以弥补python语言低性能的缺 ...

  4. Python基础笔记系列十二:requests模块的简单应用

    本系列教程供个人学习笔记使用,如果您要浏览可能需要其它编程语言基础(如C语言),why?因为我写得烂啊,只有我自己看得懂!! httpbin httpbin这个网站能测试 HTTP 请求和响应的各种信 ...

  5. Python基础笔记系列十:模块

    本系列教程供个人学习笔记使用,如果您要浏览可能需要其它编程语言基础(如C语言),why?因为我写得烂啊,只有我自己看得懂!! 模块 #1.类比于java中的jar包,模块能让你能够有逻辑地组织你的Py ...

  6. LeetCode 笔记系列13 Jump Game II [去掉不必要的计算]

    题目: Given an array of non-negative integers, you are initially positioned at the first index of the ...

  7. LeetCode 笔记系列六 Reverse Nodes in k-Group [学习如何逆转一个单链表]

    题目:Given a linked list, reverse the nodes of a linked list k at a time and return its modified list. ...

  8. LeetCode 笔记系列16.2 Minimum Window Substring [从O(N*M), O(NlogM)到O(N),人生就是一场不停的战斗]

    题目:Given a string S and a string T, find the minimum window in S which will contain all the characte ...

  9. LeetCode 笔记系列16.1 Minimum Window Substring [从O(N*M), O(NlogM)到O(N),人生就是一场不停的战斗]

    题目: Given a string S and a string T, find the minimum window in S which will contain all the charact ...

随机推荐

  1. Makefile 使用小结

    Makefile的基本格式 #目标:依赖(条件) # 命令 #all: add.c sub.c dive.c mul.c main.c # gcc add.c sub.c div.c mul.c ma ...

  2. 喵神 onevcat 的直播首秀

    喵神 onevcat 的直播首秀   王巍在圈内人称喵神,我和他在网上很早就认识,平时多有交流.在我眼中,他是一个幽默风趣高手.虽然他的博客中主要内容是 iOS 开发,但是他实际上涉及的技术领域还包括 ...

  3. 区别:Use MFC In A Shared DLL 和 Use MFC In A Static Library

    摘自:Programming Windows with MFC, 2nd Edition Choosing Use MFC In A Shared DLL minimizes your applica ...

  4. You-Get 视频下载工具 Python命令行下载工具

    You-Get 是一个命令行工具, 用来下载各大视频网站的视频, 是我目前知道的命令行下载工具中最好的一个, 之前使用过 youtube-dl, 但是 youtube-dl 吧, 下载好的视频是分段的 ...

  5. C语言 · 利息计算

    算法提高 利息计算   时间限制:1.0s   内存限制:512.0MB      编制程序完成下述任务:接受两个数,一个为用户一年期定期存款金额,一个为按照百分比格式表示的利率:程序计算一年期满 后 ...

  6. web服务器优化的一些思路

    作为一个新手(并不是菜鸟,而是像我们这样的学生),维护一个网站往往是一个很头疼的问题,尤其是动态网站,更尤其是用java写的网站. 当网站的吞吐量很小的时候你会发现服务器根本不需要维护,因为几乎没有延 ...

  7. 003杰信-在jsp页面输入数据,然后在oracle数据库中插入factory数据,当字段允许为空时要特殊处理

    本博客的内容全部来自于传智播客,特在此说明. 业务要求如下:在jsp页面(jFactoryCreate.jsp)上输入数据时,转到后台,并输入到数据库. jFactoryCreate.jsp页面:

  8. 多媒体开发之rtsp 实现rtsp over tcp/http/udp---rtsp发送

    (1) (2) (3) http://itindex.net/detail/51966-海康-rtsp-客户端 http://bbs.csdn.net/topics/390488547?page=1# ...

  9. css字体属性相关。

    出处:CSS 参考手册    http://www.w3school.com.cn/cssref/index.asp text-decoration 属性 说明:这个属性允许对文本设置某种效果,如加下 ...

  10. 学习:erlang用链表实现大容量的List或者数组。

    链表的内在实质. 效率未知,待测.