“数独”是当下炙手可热的智力游戏。一般认为它的起源是“拉丁方块”,是大数学家欧拉于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. jstree 反选,测试400条数据左右有点卡

    $("#reversecheckallmachines").on("change", function () { var checkedNodes = []; ...

  2. (数据科学学习手札83)基于geopandas的空间数据分析——geoplot篇(下)

    本文示例代码.数据及文件已上传至我的Github仓库https://github.com/CNFeffery/DataScienceStudyNotes 1 简介 在上一篇文章中我们详细学习了geop ...

  3. Python 图像处理 OpenCV (2):像素处理与 Numpy 操作以及 Matplotlib 显示图像

    前文传送门: 「Python 图像处理 OpenCV (1):入门」 普通操作 1. 读取像素 读取像素可以通过行坐标和列坐标来进行访问,灰度图像直接返回灰度值,彩色图像则返回B.G.R三个分量. 需 ...

  4. 5.6 Go 常用函数

    5.6 Go 常用函数 最正确的学习模块姿势: https://golang.org/pkg/ //golang官网 程序开发常用函数 strings处理字符串相关 统计字符串长度,按字节 len(s ...

  5. linux常用命令---终端与目录操作

    终端相关操作 目录相关操作

  6. Istio-架构

    读书笔记整理 工作机制:分为控制面和数据面 控制面:Pilot, Mixer(接收来自Envoy上报的数据), Citadel(证书和密钥管理) 数据面:Envoy 工作流程: 自动注入 应用程序启动 ...

  7. SQL——MySQL数据类型

    Text类型: Number类型: Date类型:

  8. afert和b的伪类画三角形

    /* 提示信息 */ .content-tishi{ width: 6.93rem; margin: 0 auto; background: #e9eaea; display: flex; flex- ...

  9. web自动化之键盘操作

    简单介绍下web自动化怎么触发键盘操作 按键操作 需要导入的类from selenium.webdriver.common.keys import Keys 组合键Keys.CONTROL 也就是我们 ...

  10. 剑指Offer之旋转数组的最小数字

    题目描述 把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转.输入一个非递减排序的数组的一个旋转,输出旋转数组的最小元素.例如数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转 ...