- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我尝试编写一个数独求解器,通过回溯来解决数独问题。这工作得很好,所以我尝试编写一个可以工作的 GUI,但是创建边框很困难而且很长,也许您对如何做到这一点有更好的想法。现在我希望您可以看到回溯算法的工作原理,但我不知道如何同步 Swing。
我认为 SwingWorker 在这里不起作用,因为它必须调用自身。同步块(synchronized block)不起作用。
我的代码:
package games.sudoku;
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.FlowLayout;
import java.awt.Font;
import java.awt.GridLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.beans.PropertyChangeListener;
import java.beans.PropertyChangeSupport;
import javax.swing.BorderFactory;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JTextField;
public class MyFrame {
private JFrame f = new JFrame("Sudoku Solver");
private JPanel mainPanel = new JPanel();
private JPanel sudokuPanel = new JPanel();
private JPanel buttonPanel = new JPanel();
private JTextField [][] cells = new JTextField[9][9];
private JTextField progressBar = new JTextField();
private JButton solve = new JButton("Solve!");
private JButton reset = new JButton("Reset");
private int[][] intCells = new int[9][9];
private int x = 0;
private int y = 0;
private Font sudokuFont = new Font("SansSerief", Font.BOLD, 20);
private ActionListener al;
private PropertyChangeSupport pcs = new PropertyChangeSupport(intCells);
public MyFrame() {
f.setSize(400, 400);
f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
f.add(init());
f.setResizable(false);
f.setVisible(true);
}
/**
* create and initialized mainPanel
*
* @return completly initialized mainPanel
*/
private JPanel init(){
listeners();
solve.addActionListener(al);
progressBar.setEditable(false);
// Init the panel where the sudoku is displayed
sudokuPanel.setLayout(new GridLayout(9, 9));
for(x = 0; x < cells.length; x++){
for(y = 0; y < cells[x].length; y++){
cells[x][y] = new JTextField();
cells[x][y].setFont(sudokuFont);
cells[x][y].setHorizontalAlignment(JTextField.CENTER);
//TODO find a better way
//Create borders to simulate a sudoku's layout
if(bordered(x, 0)){ // Top total
if(bordered(y, 0)){ //Left
cells[x][y].setBorder(BorderFactory.createMatteBorder(4, 4, 1, 1, Color.BLACK));
}else if(bordered(y, 1, 4, 7)){ //Mid
cells[x][y].setBorder(BorderFactory.createMatteBorder(4, 1, 1, 1, Color.BLACK));
}else if(bordered(y, 2, 5)){ //Right-Half
cells[x][y].setBorder(BorderFactory.createMatteBorder(4, 1, 1, 2, Color.BLACK));
}else if(bordered(y, 3, 6)){ //Left-Half
cells[x][y].setBorder(BorderFactory.createMatteBorder(4, 2, 1, 1, Color.BLACK));
}else{ //Right
cells[x][y].setBorder(BorderFactory.createMatteBorder(4, 1, 1, 4, Color.BLACK));
}
}else if(bordered(x, 3, 6)){ //top of box
if(bordered(y, 0)){ //Left
cells[x][y].setBorder(BorderFactory.createMatteBorder(2, 4, 1, 1, Color.BLACK));
}else if(bordered(y, 1, 4, 7)){ //Mid
cells[x][y].setBorder(BorderFactory.createMatteBorder(2, 1, 1, 1, Color.BLACK));
}else if(bordered(y, 2, 5)){ //Right-Half
cells[x][y].setBorder(BorderFactory.createMatteBorder(2, 1, 1, 2, Color.BLACK));
}else if(bordered(y, 3, 6)){ //Left-Half
cells[x][y].setBorder(BorderFactory.createMatteBorder(2, 2, 1, 1, Color.BLACK));
}else{ //Right
cells[x][y].setBorder(BorderFactory.createMatteBorder(2, 1, 1, 4, Color.BLACK));
}
}else if(bordered(x, 1, 4, 7)){ //Mid
if(bordered(y, 0)){ //Left
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 4, 1, 1, Color.BLACK));
}else if(bordered(y, 1, 4, 7)){ //Mid
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 1, 1, 1, Color.BLACK));
}else if(bordered(y, 2, 5)){ //Right-Half
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 1, 1, 2, Color.BLACK));
}else if(bordered(y, 3, 6)){ //Left-Half
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 2, 1, 1, Color.BLACK));
}else{ //Right
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 1, 1, 4, Color.BLACK));
}
}else if(bordered(x, 2, 5)){ // Bottom of box
if(bordered(y, 0)){ //Left
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 4, 2, 1, Color.BLACK));
}else if(bordered(y, 1, 4, 7)){ //Mid
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 1, 2, 1, Color.BLACK));
}else if(bordered(y, 2, 5)){ //Right-Half
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 1, 2, 2, Color.BLACK));
}else if(bordered(y, 3, 6)){ //Left-Half
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 2, 2, 1, Color.BLACK));
}else{ //Right
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 1, 2, 4, Color.BLACK));
}
}else if(bordered(x, 8)){ // Bottom overall
if(bordered(y, 0)){ //Left
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 4, 4, 1, Color.BLACK));
}else if(bordered(y, 1, 4, 7)){ //Mid
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 1, 4, 1, Color.BLACK));
}else if(bordered(y, 2, 5)){ //Right-Half
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 1, 4, 2, Color.BLACK));
}else if(bordered(y, 3, 6)){ //Left-Half
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 2, 4, 1, Color.BLACK));
}else{ //Right
cells[x][y].setBorder(BorderFactory.createMatteBorder(1, 1, 4, 4, Color.BLACK));
}
}
sudokuPanel.add(cells[x][y]);
}
}
buttonPanel.setLayout(new FlowLayout());
buttonPanel.add(solve);
buttonPanel.add(reset);
mainPanel.setLayout(new BorderLayout());
mainPanel.add(sudokuPanel, BorderLayout.CENTER);
mainPanel.add(buttonPanel, BorderLayout.SOUTH);
mainPanel.add(progressBar, BorderLayout.NORTH);
return mainPanel;
}
private void listeners(){
al = new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
if(e.getSource().equals(solve)){
for(int i = 0; i < cells.length; i++){
for(int j = 0; j < cells[i].length; j++){
if(isInt(cells[i][j].getText())){
intCells[i][j] = new Integer(cells[i][j].getText());
}else if(cells[i][j].getText().equals("")){
intCells[i][j] = 0;
cells[i][j].setForeground(Color.ORANGE);
}else{
setProgressBarText(true, "Error: Input has to be a digit between 1 and 9!");
}
}
}
pcs.firePropertyChange("input", null, null);
}else{
//TODO al for reset-button
}
}
};
}
/**
* method to stint y == 0 || y == 1 ...
*
* @param var x or y
* @param args specified numbers to check
* @return true or false if var == args
*/
private boolean bordered(int var, int... args){
for(int i = 0; i < args.length; i++){
if(var == args[i]){
return true;
}
}
return false;
}
/**
* check wether a string can be parsed to int
*
* @param s input String which should be checked
* @return false if s cannot be parsed to int or 0 < x < 10
* to check if the int can be part of the sudoku (1-9 only)
*/
private boolean isInt(String s){
try{
int i = Integer.parseInt(s);
if(i < 0 || i > 10){
return false;
}
}catch(Exception e){
return false;
}
return true;
}
/**
* set text of the progrss bar in top of the frame
*
* @param error if error --> text displayed in red, otherwise in black
* @param text text to be displayed
*/
//TODO synchronize
public void setProgressBarText(boolean error, String text){
progressBar.setText(text);
if(error){
progressBar.setForeground(Color.RED);
}else{
progressBar.setForeground(Color.BLACK);
}
}
/**
* Adds a PropertyChangeLsitener to our PropertyChangeSupport
*
* @param listener PropertyChangeListener to be added to PropertyChangeSupport
*/
public void addPropertyChangeListener(PropertyChangeListener listener) {
pcs.addPropertyChangeListener(listener);
}
/**
* display the sudoku cells
*
* @param cells two-dim int-array containing the number to be displayed
* if "0", nothing ("") is displayed
*/
//TODO synchronize
public void displaySudoku(int[][] cells){
for(int i = 0; i < cells.length; i++){
for(int j = 0; j < cells[i].length; j++){
if(cells[i][j] != 0){
this.cells[i][j].setText(Integer.toString(cells[i][j]));
}else{
this.cells[i][j].setText("");
}
}
}
}
/**
* number of tries, will be displayed in progress bar
*
* @param n ammount of tries
*/
public void setAmmountOfTries(int n){
setProgressBarText(false, Integer.toString(n) + " tries");
}
public int[][] getModel(){
return intCells;
}
}
和
package games.sudoku;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
public class SudokuSolver{
//Sudoku model to be solved
private int[][] model;
private int tries = 1;
private final MyFrame f;
public SudokuSolver() {
f = new MyFrame();
f.addPropertyChangeListener(new PropertyChangeListener() {
@Override
public void propertyChange(PropertyChangeEvent evt) {
model = f.getModel();
//TODO check if model is legally
//Try to solve
if(solve(0, 0)){
}else{
//Model is not solvable
f.setProgressBarText(true, "Sudoku cannot be solved!");
}
}
});
}
/**
* recursive method to solve the sudoku via backtracking
*
* @param row
* @param col
* @return true if a possible solution was found, otherwise false
*/
public boolean solve(int row, int col){
//Update GUI
f.displaySudoku(model);
f.setAmmountOfTries(tries);
tries++;
//Change column if all rows of it are filled
if(row == 9){
row = 0;
if(++col == 9){
return true;
}
}
//Skip non-empty cell
if(model[row][col] != 0){
return solve(row+1, col);
}
//Solve
for(int num = 1; num < 10; num++){
if(isValid(row, col, num)){
model[row][col] = num;
if(solve(row+1, col)){
return true;
}
}
}
//Reset
model[row][col] = 0;
return false;
}
/**
* check if a number can be filled in in position [row][col]
*
* @param row
* @param col
* @param num number to be filled in
* @return true if number is legal, otherwise false
*/
private boolean isValid(int row, int col, int num){
//Check row and column
for(int i = 0; i < 9; i++){
if(num == model[row][i] || num == model[i][col]){
return false;
}
}
//Check box
row = (row / 3) * 3 ;
col = (col / 3) * 3 ;
for(int r = 0; r < 3; r++){
for(int c = 0; c < 3; c++){
if(model[row+r][col+c] == num){
return false;
}
}
}
return true;
}
}
最佳答案
您似乎在 Swing 事件线程上进行长时间运行的递归调用,这将导致您的 GUI 卡住。这种情况下的解决方案是执行您似乎积极避免执行的操作:使用 SwingWorker 创建后台线程并在此后台线程中运行递归代码。同步不会有任何帮助,所以不要对那棵树吠叫。您声明不能以递归方式调用 SwingWorker,但您不必这样做;相反,您可以从 SwingWorker 内部开始递归调用,然后将其生成的数据输出到 GUI,或者通过 did 方法返回最终结果,或者通过发布/处理方法对在 SwingWorker 运行时显示临时输出。
有关此内容的更多信息,请查看 Lesson: Concurrency in Swing .
至于边框,我将在 3 x 3 GridLayout 中嵌套 9 个 JPanel,在小 JPanel 内部使用细线边框,在较小 JPanel 外部使用较粗的线边框。
例如(仅布局和边框):
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Font;
import java.awt.GridLayout;
import javax.swing.*;
public class SimpleSudokuPanel extends JPanel {
private static final int PANEL_THICKNESS = 2;
private static final int TF_THICKNESS = 1;
private static final int TF_COLS = 2;
private static final float TF_PTS = 36f;
private JTextField[][] grid = new JTextField[9][9];
public SimpleSudokuPanel() {
JPanel mainPanel = new JPanel(new GridLayout(3, 3));
JPanel[] innerPanels = new JPanel[9];
for (int i = 0; i < innerPanels.length; i++) {
innerPanels[i] = new JPanel(new GridLayout(3, 3));
innerPanels[i].setBorder(BorderFactory.createLineBorder(Color.black, PANEL_THICKNESS));
mainPanel.add(innerPanels[i]);
}
for (int i = 0; i < grid.length; i++) {
for (int j = 0; j < grid[i].length; j++) {
grid[i][j] = new JTextField(TF_COLS);
grid[i][j].setFont(grid[i][j].getFont().deriveFont(Font.BOLD, TF_PTS));
grid[i][j].setBorder(BorderFactory.createLineBorder(Color.black, TF_THICKNESS));
grid[i][j].setHorizontalAlignment(JTextField.CENTER);
int panelIndex = 3 * (i / 3) + j / 3;
innerPanels[panelIndex].add(grid[i][j]);
}
}
setBorder(BorderFactory.createLineBorder(Color.black, PANEL_THICKNESS));
setLayout(new BorderLayout());
add(mainPanel, BorderLayout.CENTER);
}
private static void createAndShowGui() {
JFrame frame = new JFrame("SimpleSudokuPanel");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.getContentPane().add(new SimpleSudokuPanel());
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
public static void main(String[] args) {
SwingUtilities.invokeLater(() -> createAndShowGui());
}
}
关于java - 如何在递归方法内同步 Swing,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37618515/
我正在编写一个具有以下签名的 Java 方法。 void Logger(Method method, Object[] args); 如果一个方法(例如 ABC() )调用此方法 Logger,它应该
我是 Java 新手。 我的问题是我的 Java 程序找不到我试图用作的图像文件一个 JButton。 (目前这段代码什么也没做,因为我只是得到了想要的外观第一的)。这是我的主课 代码: packag
好的,今天我在接受采访,我已经编写 Java 代码多年了。采访中说“Java 垃圾收集是一个棘手的问题,我有几个 friend 一直在努力弄清楚。你在这方面做得怎么样?”。她是想骗我吗?还是我的一生都
我的 friend 给了我一个谜语让我解开。它是这样的: There are 100 people. Each one of them, in his turn, does the following
如果我将使用 Java 5 代码的应用程序编译成字节码,生成的 .class 文件是否能够在 Java 1.4 下运行? 如果后者可以工作并且我正在尝试在我的 Java 1.4 应用程序中使用 Jav
有关于why Java doesn't support unsigned types的问题以及一些关于处理无符号类型的问题。我做了一些搜索,似乎 Scala 也不支持无符号数据类型。限制是Java和S
我只是想知道在一个 java 版本中生成的字节码是否可以在其他 java 版本上运行 最佳答案 通常,字节码无需修改即可在 较新 版本的 Java 上运行。它不会在旧版本上运行,除非您使用特殊参数 (
我有一个关于在命令提示符下执行 java 程序的基本问题。 在某些机器上我们需要指定 -cp 。 (类路径)同时执行java程序 (test为java文件名与.class文件存在于同一目录下) jav
我已经阅读 StackOverflow 有一段时间了,现在我才鼓起勇气提出问题。我今年 20 岁,目前在我的家乡(罗马尼亚克卢日-纳波卡)就读 IT 大学。足以介绍:D。 基本上,我有一家提供簿记应用
我有 public JSONObject parseXML(String xml) { JSONObject jsonObject = XML.toJSONObject(xml); r
我已经在 Java 中实现了带有动态类型的简单解释语言。不幸的是我遇到了以下问题。测试时如下代码: def main() { def ks = Map[[1, 2]].keySet()
一直提示输入 1 到 10 的数字 - 结果应将 st、rd、th 和 nd 添加到数字中。编写一个程序,提示用户输入 1 到 10 之间的任意整数,然后以序数形式显示该整数并附加后缀。 public
我有这个 DownloadFile.java 并按预期下载该文件: import java.io.*; import java.net.URL; public class DownloadFile {
我想在 GUI 上添加延迟。我放置了 2 个 for 循环,然后重新绘制了一个标签,但这 2 个 for 循环一个接一个地执行,并且标签被重新绘制到最后一个。 我能做什么? for(int i=0;
我正在对对象 Student 的列表项进行一些测试,但是我更喜欢在 java 类对象中创建硬编码列表,然后从那里提取数据,而不是连接到数据库并在结果集中选择记录。然而,自从我这样做以来已经很长时间了,
我知道对象创建分为三个部分: 声明 实例化 初始化 classA{} classB extends classA{} classA obj = new classB(1,1); 实例化 它必须使用
我有兴趣使用 GPRS 构建车辆跟踪系统。但是,我有一些问题要问以前做过此操作的人: GPRS 是最好的技术吗?人们意识到任何问题吗? 我计划使用 Java/Java EE - 有更好的技术吗? 如果
我可以通过递归方法反转数组,例如:数组={1,2,3,4,5} 数组结果={5,4,3,2,1}但我的结果是相同的数组,我不知道为什么,请帮助我。 public class Recursion { p
有这样的标准方式吗? 包括 Java源代码-测试代码- Ant 或 Maven联合单元持续集成(可能是巡航控制)ClearCase 版本控制工具部署到应用服务器 最后我希望有一个自动构建和集成环境。
我什至不知道这是否可能,我非常怀疑它是否可能,但如果可以,您能告诉我怎么做吗?我只是想知道如何从打印机打印一些文本。 有什么想法吗? 最佳答案 这里有更简单的事情。 import javax.swin
我是一名优秀的程序员,十分优秀!