7.4 dfs例题-----数独游戏

题目:

即每一行、每一列、每一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();
		}
		
	}
	
}


 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值