“数独”是当下炙手可热的智力游戏。一般认为它的起源是“拉丁方块”,是大数学家欧拉于1783年发明的。

如图[1.jpg]所示:6x6的小格被分为6个部分(图中用不同的颜色区分),每个部分含有6个小格(以下也称为分组)。

开始的时候,某些小格中已经填写了字母(ABCDEF之一)。需要在所有剩下的小格中补填字母。

全部填好后,必须满足如下约束:

1. 所填字母只允许是A,B,C,D,E,F 中的某一个。

2. 每行的6个小格中,所填写的字母不能重复。

3. 每列的6个小格中,所填写的字母不能重复。

4. 每个分组(参见图中不同颜色表示)包含的6个小格中,所填写的字母不能重复。

为了表示上的方便,我们用下面的6阶方阵来表示图[1.jpg]对应的分组情况(组号为0~5):

022013

243333

445555

用下面的数据表示其已有字母的填写情况:

02C

03B

05A

20D

35E

53F

很明显,第一列表示行号,第二列表示列号,第三列表示填写的字母。行号、列号都从0开始计算。

一种可行的填写方案(此题刚好答案唯一)为:

E F C B D A

A C E D F B

D A B E C F

F B D C A E

B D F A E C

C E A F B D

你的任务是:编写程序,对一般的拉丁方块问题求解,如果多解,要求找到所有解。

【输入、输出格式要求】

用户首先输入6行数据,表示拉丁方块的分组情况。

接着用户输入一个整数n (n<36), 表示接下来的数据行数

接着输入n行数据,每行表示一个预先填写的字母。

程序则输出所有可能的解(各个解间的顺序不重要)。

每个解占用7行。

即,先输出一个整数,表示该解的序号(从1开始),接着输出一个6x6的字母方阵,表示该解。

解的字母之间用空格分开。

如果找不到任何满足条件的解,则输出“无解”

例如:用户输入:

022013

243333

445555

02C

03B

05A

20D

35E

53F

则程序输出:

E F C B D A

A C E D F B

D A B E C F

F B D C A E

B D F A E C

C E A F B D

再如,用户输入:

002113

022443

555553

04B

05A

13D

14C

24E

50C

51A

则程序输出:

D C E F B A

E F A D C B

A B F C E D

B E D A F C

F D C B A E

C A B E D F

D C E F B A

E F A D C B

A D F B E C

B E C A F D

F B D C A E

C A B E D F

D C F E B A

A E B D C F

F D A C E B

B F E A D C

E B C F A D

C A D B F E

D C F E B A

B E A D C F

A D C F E B

F B E A D C

E F B C A D

C A D B F E

D C F E B A

E F A D C B

A B C F E D

B E D A F C

F D B C A E

C A E B D F

D C F E B A

E F A D C B

A B D F E C

B E C A F D

F D B C A E

C A E B D F

D C F E B A

E F A D C B

A D B F E C

B E C A F D

F B D C A E

C A E B D F

D C F E B A

F E A D C B

A D B C E F

B F E A D C

E B C F A D

C A D B F E

D C F E B A

F E A D C B

A F C B E D

B D E A F C

E B D C A F

C A B F D E

package com.liu.ex9;

import java.util.HashSet;
import java.util.Scanner; public class Main {
public static int[][] group = new int[6][6]; //输入分组情况
public static char[][] result = new char[6][6]; //满足题意的填充结果
public static int[][] row = new int[6][6]; //检测6行是否均为不同字母
public static int[][] col = new int[6][6]; //检测6列是否均为不同字母
public static int[][] set = new int[6][6]; //检测0~5组是否均为不同字母
public static char[] value = {'A','B','C','D','E','F'}; //填充字母
public static int count = 0; //统计最终解个数
public HashSet<String> hash = new HashSet<String>(); public boolean check() {
StringBuilder temp = new StringBuilder("");
for(int i = 0;i < 6;i++) {
for(int j = 0;j < 6;j++)
temp.append(result[i][j]+" ");
temp.append("\n");
}
String A = temp.toString();
if(hash.contains(A))
return false;
return true;
} public void dfs(int step) {
if(step >= 37) {
if(check()) { //检测是否有重复解
count++;
System.out.println(count);
StringBuilder temp = new StringBuilder("");
for(int i = 0;i < 6;i++) {
for(int j = 0;j < 6;j++)
temp.append(result[i][j]+" ");
temp.append("\n");
}
hash.add(temp.toString());
System.out.print(temp);
}
return;
} else {
int tempRow = (step - 1) / 6;
int tempCol = (step - 1) % 6;
int tempSet = group[tempRow][tempCol];
for(int i = 0;i < 6;i++) {
if(result[tempRow][tempCol] == '-') {
char temp = (char) ('A' + i);
if(row[tempRow][i] == 0 && col[tempCol][i] == 0 && set[tempSet][i] == 0) {
result[tempRow][tempCol] = temp;
row[tempRow][i] = 1;
col[tempCol][i] = 1;
set[tempSet][i] = 1;
dfs(step + 1);
result[tempRow][tempCol] = '-'; //回溯处理
row[tempRow][i] = 0;
col[tempCol][i] = 0;
set[tempSet][i] = 0; } else {
continue;
}
} else {
dfs(step + 1);
}
}
}
return;
} public static void main(String[] args) {
Main test = new Main();
for(int i = 0;i < 6;i++)
for(int j = 0;j < 6;j++)
result[i][j] = '-'; //初始化为填充格子字符为'-'
Scanner in = new Scanner(System.in);
for(int i = 0;i < 6;i++) {
String temp = in.nextLine();
for(int j = 0;j < 6;j++)
group[i][j] = temp.charAt(j) - '0';
}
int n = in.nextInt();
in.nextLine();
for(int i = 0;i < n;i++) {
String temp = in.nextLine();
int a = temp.charAt(0) - '0';
int b = temp.charAt(1) - '0';
int v = temp.charAt(2) - 'A';
result[a][b] = temp.charAt(2);
row[a][v] = 1; //表示第a行位于第v个的位置,已经填充
col[b][v] = 1; //表示第b列位于第v个的位置,已经填充
int tempSet = group[a][b]; //获取(a,b)小组组号
set[tempSet][v] = 1; //表示第tempSet小组第v个位置,已经填充
}
test.dfs(1);
}
}

java实现拉丁方块填数字的更多相关文章

  1. java实现第三届蓝桥杯方块填数

    方块填数 "数独"是当下炙手可热的智力游戏.一般认为它的起源是"拉丁方块",是大数学家欧拉于1783年发明的. 如图[1.jpg]所示:6x6的小格被分为6个部 ...

  2. TOJ 1220 填数字游戏 / 深搜

    填数字游戏 时间限制(普通/Java):1000MS/10000MS     运行内存限制:65536KByte 描述 有个小游戏,让你填写以下方框,要求: a1+a2+a3+M=b1+b2+b3+M ...

  3. ACM/ICPC 之 DP进阶(51Nod-1371(填数字))

    原题链接:填数字 顺便推荐一下,偶然看到这个OJ,发现社区运营做得很赞,而且交互和编译环境都很赞(可以编译包括Python,Ruby,Js在内的脚本语言,也可以编译新标准的C/C++11,甚至包括Go ...

  4. 比较java与c语言中数字转换成字符的不同

    java java中将数字转换成字符非常方便,只要用一个"+"然后在跟一个空格行了.比如,你输入一个122 ,就会变成"122 ". import java.u ...

  5. java截取字符串中的数字

    java从字符串中提取数字 随便给你一个含有数字的字符串,比如: String s="eert343dfg56756dtry66fggg89dfgf"; 那我们如何把其中的数字提取 ...

  6. Java中用正则表达式找出数字

    Java中用正则表达式找出数字 1.题目    String str = "fjd789klsd908434jk#$$%%^38488545",从中找出78990843438488 ...

  7. Java基于opencv实现图像数字识别(五)—投影法分割字符

    Java基于opencv实现图像数字识别(五)-投影法分割字符 水平投影法 1.水平投影法就是先用一个数组统计出图像每行黑色像素点的个数(二值化的图像): 2.选出一个最优的阀值,根据比这个阀值大或小 ...

  8. Java基于opencv实现图像数字识别(四)—图像降噪

    Java基于opencv实现图像数字识别(四)-图像降噪 我们每一步的工作都是基于前一步的,我们先把我们前面的几个函数封装成一个工具类,以后我们所有的函数都基于这个工具类 这个工具类呢,就一个成员变量 ...

  9. Java基于opencv实现图像数字识别(三)—灰度化和二值化

    Java基于opencv实现图像数字识别(三)-灰度化和二值化 一.灰度化 灰度化:在RGB模型中,如果R=G=B时,则彩色表示灰度颜色,其中R=G=B的值叫灰度值:因此,灰度图像每个像素点只需一个字 ...

随机推荐

  1. Elasticsearch系列---几个高级功能

    概要 本篇主要介绍一下搜索模板.映射模板.高亮搜索和地理位置的简单玩法. 标准搜索模板 搜索模板search tempalte高级功能之一,可以将我们的一些搜索进行模板化,使用现有模板时传入指定的参数 ...

  2. 关于layui数据表格的各种事件

    table.on('tool(demo)', function(obj){}):监听工具条事件,tool 是工具条事件名,demo 是 table 原始容器的属性 lay-filter="对 ...

  3. Python单元测试框架:unittest(一)

    Python单元测试框架unittest使用方法讲解 主要介绍了Python单元测试框架unittest使用方法讲解,本文讲解了unittest概述.命令行接口.测试案例自动搜索.创建测试代码.构建测 ...

  4. 分布式文件存储库MinIO可还行?

    在传统的单体应用架构中,一个应用程序对应一台服务器,提供单进程服务. 但是随着业务的升级,技术的更新迭代,分布式.集群架构.微服务等现已俨然成为主流. 几乎所有的项目都会与文件挂钩,例如OA系统的报表 ...

  5. Redux:with React(一)

    作者数次强调,redux和React没有关系(明明当初就是为了管理react的state才弄出来的吧),它可以和其他插件如 Angular, Ember, jQuery一起使用.好啦好啦知道啦.Red ...

  6. scrapy框架简介与安装启动

    Scrapy 是一个专业的.高效的爬虫框架,它使用专业的 Twisted 包(基于事件驱动的网络引擎包)高效地处理网络通信,使用 lxml(专业的 XML 处理包).cssselect 高效地提取 H ...

  7. redis 主从哨兵01

    主从复制过程 1.从服务器开始连接主服务器时,会向主服务器发送一个SYNC同步命令 2.主服务器接收到命令后,执行BGSAVE,异步的将写命令保存到一个缓冲区里 3.主服务器执行完BGSAVE之后,就 ...

  8. 阿里云服务器 ECS Ubuntu系统下PHP,MYSQL,APACHE2的安装配置

    1.系统更新,必须更新,否则有些软件会找不到. apt-get update apt-get upgrade 2.安装mysql sudo apt-get install mysql-server 3 ...

  9. ios审核 "prefs:root="被拒

    https://blog.csdn.net/xnickname666/article/details/83068516 使用TZImagePicker  https://github.com/banc ...

  10. UVALive8518 Sum of xor sum

    题目链接:https://vjudge.net/problem/UVALive-8518 题目大意: 给定一个长度为 $N$ 的数字序列 $A$,进行 $Q$ 次询问,每次询问 $[L,R]$,需要回 ...