gpt4 book ai didi

java - 数独解算器错误

转载 作者:塔克拉玛干 更新时间:2023-11-03 05:22:34 25 4
gpt4 key购买 nike

我不知道自己做错了什么,整天盯着这段代码看。这是 Java 中的“标准”数独解算器,它采用 int[][],其中空白处为 0。鉴于我只传递了一 block 有 35 个孔的板,这应该能够解决绝大多数问题,但只能解决 ~66%。在其他情况下,还剩下一些(通常是 2 或 4 个)空白,无法解决(即,不正确的数字已写入 board。)几乎总是,它会是一个9 个不见了。

我明白这样一个简单的解决方案并不能解决所有的数独问题。我故意给它简单的。

import java.util.ArrayList;
import java.util.List;

public class SudokuSolver
{
public SudokuSolver()
{
init();
}

public boolean solve()
{
/* Each method checks (in different ways) to see if it can find a new number
If said method does find a number, it sets off a chain reaction, starting back at the beginning.
*/
int countdown = 20;
while(!solved() && --countdown > 0)
{
if(given())
continue;
if(findSingletons())
continue;
if(zerosLeft() <= 4)
justGuess();
}
return solved();
}

public boolean given()
{
boolean repeat = false;
//Iterate through every given number
for(int i=0;i<9;i++)
{
for(int j=0;j<9;j++)
{
if(board[i][j] != 0 && !found[i][j])
{
repeat = true;
foundNum(i, j, board[i][j]);
}
}
}
//Call given every time a new number is found
return repeat;
}

public boolean findSingletons()
{
boolean repeat = false;
//LOTS of iteration, but I'm out of ideas.
int[] values;
ArrayList<Integer> singletons = new ArrayList<Integer>();
for(int i=0;i<9;i++)
{
values = new int[10];
singletons.clear();
for(int j=0;j<9;j++)
for(int k=0;k<possible[i][j].size();k++)
values[possible[i][j].get(k)]++;
for(int j=1;j<10;j++)
if(values[j] == 1)
singletons.add(j);
for(int j=0;j<9;j++)
for(int k=0;k<singletons.size();k++)
if(possible[i][j].contains(singletons.get(k)))
{
foundNum(i, j, singletons.get(k));
repeat = true;
}
}

for(int i=0;i<9;i++)
{
values = new int[10];
singletons.clear();
for(int j=0;j<9;j++)
for(int k=0;k<possible[j][i].size();k++)
values[possible[j][i].get(k)]++;
for(int j=1;j<10;j++)
if(values[j] == 1)
singletons.add(j);
for(int j=0;j<9;j++)
for(int k=0;k<singletons.size();k++)
if(possible[j][i].contains(singletons.get(k)))
{
foundNum(j, i, singletons.get(k));
repeat = true;
}
}

int[] corners = {0,3,6};
for(int a=0;a<3;a++)
for(int l=0;l<3;l++)
for(int i=corners[a];i<corners[a]+3;i++)
{
values = new int[10];
singletons.clear();
for(int j=corners[l];j<corners[l]+3;j++)
for(int k=0;k<possible[i][j].size();k++)
values[possible[i][j].get(k)]++;
for(int j=1;j<10;j++)
if(values[j] == 1)
singletons.add(j);
for(int j=0;j<9;j++)
for(int k=0;k<singletons.size();k++)
if(possible[i][j].contains(singletons.get(k)))
{
foundNum(i, j, singletons.get(k));
repeat = true;
}
}
return repeat;
}

public void justGuess()
{
outer:
for(int i=0;i<9;i++)
for(int j=0;j<9;j++)
if(board[i][j] == 0)
{
foundNum(i, j, possible[i][j].get(0));
break outer;
}
}

public void foundNum(int x, int y, int numFound)
{

if(board[x][y] != 0 && board[x][y] != numFound)
{
throw new RuntimeException("Attempting to place a number where one was already found");
}

board[x][y] = numFound;
possible[x][y].clear();
possible[x][y].add(numFound);
found[x][y] = true;

for(int i=0;i<9;i++) {
if(i != x)
if(possible[i][y].indexOf(numFound) != -1)
possible[i][y].remove(possible[i][y].indexOf(numFound));
}
for(int i=0;i<9;i++) {
if(i != y)
if(possible[x][i].indexOf(numFound) != -1)
possible[x][i].remove(possible[x][i].indexOf(numFound));
}
int cornerX = 0;
int cornerY = 0;
if(x > 2)
if(x > 5)
cornerX = 6;
else
cornerX = 3;
if(y > 2)
if(y > 5)
cornerY = 6;
else
cornerY = 3;
for(int i=cornerX;i<10 && i<cornerX+3;i++)
for(int j=cornerY;j<10 && j<cornerY+3;j++)
if(i != x && j != y)
if(possible[i][j].indexOf(numFound) != -1)
possible[i][j].remove(possible[i][j].indexOf(numFound));
}

public boolean solved() {
for(int i=0;i<9;i++)
for(int j=0;j<9;j++)
if(!found[i][j])
return false;
return true;
}

public void reset(int[][] board)
{
this.board = board;
init();
}

public void init()
{
possible = new ArrayList[9][9];
for(int i=0;i<9;i++)
for(int j=0;j<9;j++)
{
possible[i][j] = new ArrayList<Integer>();
for(int k=1;k<10;k++)
possible[i][j].add(k);
}
found = new boolean[9][9];
}

public void print()
{
for(int i=0;i<9;i++)
{
if(i%3==0 && i != 0)
System.out.println("- - - | - - - | - - -");
for(int j=0;j<9;j++)
{
if(j%3==0 & j != 0)
System.out.print("| ");
System.out.print(board[i][j] + " ");
}
System.out.println();
}
System.out.println();
}

private int zerosLeft()
{
int empty = 0;
for(int i=0;i<9;i++)
for(int j=0;j<9;j++)
if(board[i][j] == 0)
empty++;
return empty;
}

private void data(int difficulty)
{
int empty = 0;
for(int i=0;i<9;i++)
for(int j=0;j<9;j++)
if(board[i][j] == 0)
empty++;
System.out.println(empty);
}

public static void main(String[] args)
{
SudokuGenerator sg = new SudokuGenerator();
SudokuSolver ss = new SudokuSolver();
int[][] tempBoard = {{4, 0, 1, 0, 9, 7, 0, 5, 8 },
{2, 0, 0, 5, 3, 1, 4, 0, 6 },
{5, 0, 6, 4, 0, 2, 0, 3, 9 },
{0, 9, 0, 0, 0, 4, 3, 0, 2 },
{0, 0, 0, 9, 0, 0, 6, 4, 7 },
{7, 0, 4, 0, 0, 0, 9, 0, 5 },
{0, 0, 7, 0, 0, 3, 8, 9, 4 },
{8, 5, 0, 1, 4, 9, 7, 0, 0 },
{9, 0, 3, 8, 7, 6, 0, 0, 0 }};
ss.reset(tempBoard);
System.out.println(ss.solve());
ss.print();
ss.data(35);
}

int[][] board;
ArrayList<Integer>[][] possible;
boolean[][] found;
}

我对编程还是个新手,所以除了解决这个问题之外的任何建议都将受到欢迎。 (特别优化可能。这是我迄今为止编写的最亵渎的代码。)

谢谢!

最佳答案

我开始阅读您的代码,但感觉比应该的要长,而且那些循环变得非常困惑。没有什么会立即跳到我身上。您确实说过您不仅想要解决方案,还想要建议。

您必须弄清楚问题是否出在您的设计上(它不适用于解决数独问题),或者是否只是实现中某处存在一个简单的错误。也许通过并写下关于每个循环正在完成的事情的评论,“橡皮鸭测试”,被迫解释一切,你会停下来并意识到有些事情是不必要的,或者不是它需要的。这有助于解决设计问题。

如果问题是实现,你知道如何正式调试应用程序吗?设置断点并按指令遍历它?如果你有一个小错误,但你看不到哪里,那就是要走的路。找到一个失败的非常简单的示例案例,然后运行该测试并在开始时中断它。逐步执行,并遵循逻辑。希望您会看到哪里出错了。编写 JUnit 测试或日志语句很棒,但是当您遇到棘手的错误时,您必须进行一些真正的断点调试。

你的总体框架很好,你有一些对象来保存数据,还有一个很好的干净的解决方法,它调用了几个不同的方法并循环遍历它们。但是这些方法中的每一个,哇,它们肯定很乱。那种代码,大量使用相同变量名的紧密循环,大量数组操作,很容易弄错某些东西并出现错误,这使得阅读和查找错误变得非常困难。

Eclipse 使调试 java 变得非常容易,如果您以前没有调试过的话。谷歌上有很多好的教程,所以我不会打扰^_~

关于java - 数独解算器错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10368846/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com