题目:
即每一行、每一列、每一3*3的格子都含有1~9的全部数字,且不重复
解析:递归求解。
每次确定一个位置的数字,就一路走到底或者中间数字不合法,再返回上一层继续查找
import java.util.Scanner;
public class Main {
/*
* 测试数据:
800000000
003600000
070090200
050007000
000045700
000100030
001000068
008500010
090000400
* */
static char[][] table = new char[9][9];
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
for(int i=0;i<9;i++) {
table[i] = sc.nextLine().toCharArray();
}
dfs(0,0);//从table[0][0]开始寻找
}
/*一、深搜确定表格中每个位置的元素
* (1)递归出口:如果x==9,说明已经确定完所有位置上的元素(0<=x<9)
* (2)如果table[x][y]='0',该位置没有填入数字,确定其值
* 否则该位置已填入数字,继续搜索下一个位置
* */
private static void dfs(int x, int y) {
if(x==9) {//因为只有一个解,找到该解后,立即退出
print(table);
System.exit(0);//退出该函数
}
if(table[x][y]=='0') {//(1)该位置未填数字
for(int i=1;i<=9;i++) {//(1.1)遍历1~9,寻找合法的数字
if(check(x,y,i)) {//(1.2)该数字合法,将其加入该位置
table[x][y]=(char) ('0'+i);
dfs(x+(y+1)/9,(y+1)%9);//(1.3)继续向下搜索
}
}//for
table[x][y]='0';//回溯
}else {//(2)该位置已存在数字,继续向下搜索
dfs(x+(y+1)/9,(y+1)%9);
}
}
/*
* 二、判断在table[x][y]出填入数字num是否合法
* 合法需满足(1)在第x行未出现该数字
* (2)在第y列未出现该数字
* (3)在table[x][y]所在的3*3表格中,未出现该数字
* */
private static boolean check(int x, int y, int num) {
//(1)检查同行x和同列y
for(int i=0;i<9;i++) {
if((table[x][i]-'0')==num) return false;//该行已存在数字num
if((table[i][y]-'0')==num) return false;//该列已存在数字num
}
//(2)检查3*3表格是否存在数字num。下标根据x/3 x%3 y/3 y%3来确定
for(int i=(x/3)*3;i<(x/3)*3+3;i++) {
for(int j=(y/3)*3;j<(y/3)*3+3;j++) {
if(table[i][j]-'0'==num) return false;
}
}
return true;
}
/*
* 三、打印二维数组
* */
private static void print(char[][] t) {
for(int i=0;i<t.length;i++) {
for(int j=0;j<t[0].length;j++) {
System.out.print(t[i][j]);
}
System.out.println();
}
}
}