LC 37.解数独

本文介绍了一种使用回溯算法解决数独问题的Java程序,通过初始化行、列和3x3宫的记录数组,实现对数独空格逐个尝试数字并检查是否符合规则的过程,最终找到唯一解。
摘要由CSDN通过智能技术生成

37.解数独

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

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

  1. 数字 1-9 在每一行只能出现一次。
  2. 数字 1-9 在每一列只能出现一次。
  3. 数字 1-9 在每一个以粗实线分隔的 3x3 宫内只能出现一次。(请参考示例图)

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

示例 1:

输入: 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”]]
输出:[[“5”,“3”,“4”,“6”,“7”,“8”,“9”,“1”,“2”],[“6”,“7”,“2”,“1”,“9”,“5”,“3”,“4”,“8”],[“1”,“9”,“8”,“3”,“4”,“2”,“5”,“6”,“7”],[“8”,“5”,“9”,“7”,“6”,“1”,“4”,“2”,“3”],[“4”,“2”,“6”,“8”,“5”,“3”,“7”,“9”,“1”],[“7”,“1”,“3”,“9”,“2”,“4”,“8”,“5”,“6”],[“9”,“6”,“1”,“5”,“3”,“7”,“2”,“8”,“4”],[“2”,“8”,“7”,“4”,“1”,“9”,“6”,“3”,“5”],[“3”,“4”,“5”,“2”,“8”,“6”,“1”,“7”,“9”]]
解释: 输入的数独如上图所示,唯一有效的解决方案如下所示:

提示:

  • b o a r d . l e n g t h = = 9 board.length == 9 board.length==9
  • b o a r d [ i ] . l e n g t h = = 9 board[i].length == 9 board[i].length==9
  • board[i][j] 是一位数字或者 '.'
  • 题目数据 保证 输入数独仅有一个解

解法一(回溯)

思路分析:

  1. 解数独问题和N皇后问题有点类似,考虑使用回溯算法进行求解
  2. 思考回溯函数三要素:
    1. 思考回溯函数的返回值和参数类型;因为解数独只存在一种解法,所以只需找到一种便可以结束整个函数,所以返回值类型为boolean型,当找到一种解法时,返回true结束函数,参数有:board数组,其余参数再需要时进行添加。
    2. 思考回溯函数的结束条件,并不需要特意结束函数,当数独的空填完后,便不会再继续填空,函数自然结束
    3. 思考回溯函数的遍历,即使用双重for循环,遍历整个字符数组,当遍历到一个可以填空的位置时,遍历数字1-9,并判断某数字是否可以填在该空,不可以则跳过,若可以,则标记该树,并继续填空,判断是否能解开数独,若能,则结束函数,若不能,则回溯,填写新的数字。

实现代码如下:

class Solution {
	boolean[][] rows;	// 记录第i行 已放的数字
	boolean[][] cols;	// 记录第j列 已存放的数字
	boolean[][] blocks;	// 记录第几个方块 已存放的数字; 方块序号为 从左到右 从上到下
  public void solveSudoku(char[][] board) {
	// 初始化 rows、cols、blocks
	rows = new boolean[board.length][board[0].length + 1];
	cols = new boolean[board.length][board[0].length + 1];
	blocks = new boolean[board.length][board[0].length + 1];
	init(board);	// 对三个记录数组进行初始化
	backtracking(board);
  }
	private void init(char[][] board) {
		// 初始化
		for (int i = 0; i < board.length; ++ i) {
			for (int j = 0; j < board[0].length; ++ j) {
				if (board[i][j] != '.') {
					rows[i][board[i][j] - '0'] = true;	// 记录第i行
					blocks[getBlock(i, j)][board[i][j] - '0'] = true;	// 记录第几个3*3宫内
				}
				if (board[j][i] != '.') {
					cols[i][board[j][i] - '0'] = true;	// 记录第i列
				}
			}
		}
	}
	private boolean backtracking(char[][] board) {
		for (int i = 0; i < board.length; ++ i) {
			for (int j = 0; j < board[0].length; ++ j) {
				if (board[i][j] == '.') {
					for (int k = 1; k <= 9; ++ k) {
						if (isValid(i, j, k)) {
							board[i][j] = (char) (k+'0');
							rows[i][k] = true;
							cols[j][k] = true;
							blocks[getBlock(i, j)][k] = true;
							if (backtracking(board))
								return true;
							board[i][j] = '.';
							rows[i][k] = false;
							cols[j][k] = false;
							blocks[getBlock(i, j)][k] = false;
						}
					}
					return false;
				}
			}
		}
		return true;
	}
	private boolean isValid(int i, int j, int k) {
		return !(rows[i][k] || cols[j][k] || blocks[getBlock(i, j)][k]);
	}
	private int getBlock(int i, int j) {	// 根据方块坐标 获取其位于第几个3*3宫内
		return 3 * (i/3) + j/3;
	}
}

提交结果如下:

解答成功:
执行耗时:3 ms,击败了70.53% 的Java用户
内存消耗:40.1 MB,击败了32.62% 的Java用户

  • 8
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值