使用Thread.sleep()时,我遇到了JTextField.setText()的问题。这是我正在制作的基本计算器。当输入字段中的输入格式不正确时,我希望“INPUT ERROR”在输出字段中出现5秒钟,然后清除它。当我将文本设置为“INPUT ERROR”时,setText()方法 工作并且通过打印文本介于两者之间我发现它确实适用于它和setText(“”)在另一个之后。当我在它们之间放置Thread.sleep()时会出现问题。 这是代码的SSCCE版本:
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.regex.Pattern;
import javax.swing.*;
public class Calc {
static Calc calc = new Calc();
public static void main(String args[]) {
GUI gui = calc.new GUI();
}
public class GUI implements ActionListener {
private JButton equals;
private JTextField inputField, outputField;
public GUI() {
createFrame();
}
public void createFrame() {
JFrame baseFrame = new JFrame("Calculator");
baseFrame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
JPanel contentPane = new JPanel();
BoxLayout layout = new BoxLayout(contentPane, BoxLayout.Y_AXIS);
contentPane.setLayout(layout);
baseFrame.setContentPane(contentPane);
baseFrame.setSize(320, 100);
equals = new JButton("=");
equals.addActionListener(this);
inputField = new JTextField(16);
inputField.setHorizontalAlignment(JTextField.TRAILING);
outputField = new JTextField(16);
outputField.setHorizontalAlignment(JTextField.TRAILING);
outputField.setEditable(false);
contentPane.add(inputField);
contentPane.add(outputField);
contentPane.add(equals);
contentPane.getRootPane().setDefaultButton(equals);
baseFrame.setResizable(false);
baseFrame.setLocation(100, 100);
baseFrame.setVisible(true);
}
/**
* When an action event takes place, the source is identified and the
* appropriate action is taken.
*/
@Override
public void actionPerformed(ActionEvent e) {
if (e.getSource() == equals) {
inputField.setText(inputField.getText().replaceAll("\\s", ""));
String text = inputField.getText();
System.out.println(text);
Pattern equationPattern = Pattern.compile("[\\d(][\\d-+*/()]+[)\\d]");
boolean match = equationPattern.matcher(text).matches();
System.out.println(match);
if (match) {
// Another class calculates
} else {
try {
outputField.setText("INPUT ERROR"); // This doesn't appear
Thread.sleep(5000);
outputField.setText("");
} catch (InterruptedException e1) {
}
}
}
}
}
}
我实际上并没有使用嵌套类,但我希望它能够包含在一个类中。很抱歉GUI看起来如何,但这又是为了减少代码。重要部分(if (e.getSource() == equals)
)与我的代码保持不变。提供错误输入的最简单方法是使用字母。
答案 0 :(得分:12)
当您使用Thread.sleep()
时,您正在主线程上执行此操作。这会冻结gui五秒钟,然后更新outputField
。当发生这种情况时,它会使用最后一个空白的文本。
使用Swing Timers要好得多,这是一个做你想要完成的事情的例子:
if (match) {
// Another class calculates
} else {
outputField.setText("INPUT ERROR");
ActionListener listener = new ActionListener(){
public void actionPerformed(ActionEvent event){
outputField.setText("");
}
};
Timer timer = new Timer(5000, listener);
timer.setRepeats(false);
timer.start();
}
答案 1 :(得分:8)
你在Swing主线程中做了Thread.sleep()
。这不是一个好习惯。您最多需要使用SwingWorker
个帖子。
正在发生的事情是它正在运行第一行,点击Thread.sleep()
。
这可以防止(主)EDT线程执行任何重绘(以及阻止下一行执行)。
您应该使用javax.swing.Timer
设置延迟反应,而不是在主线程中调用sleep()
。
答案 2 :(得分:8)
正如菲利普怀特豪斯在答案中所述,你正在通过Thread.sleep(...)
电话阻止摆动事件派遣线程。
鉴于您已经花时间设置ActionListener
,最好使用javax.swing.Timer
来控制清除文本。为此,您可以在GUI
类中添加一个字段:
private Timer clearTimer = new Timer(5000, this);
在GUI
的构造函数中,关闭重复功能,因为您实际上只需要一次性:
public GUI() {
clearTimer.setRepeats(false);
createFrame();
}
然后,可以修改actionPerformed
以使用它来启动计时器/清除字段:
public void actionPerformed(ActionEvent e) {
if (e.getSource() == equals) {
inputField.setText(inputField.getText().replaceAll("\\s", ""));
String text = inputField.getText();
System.out.println(text);
Pattern equationPattern = Pattern.compile("[\\d(][\\d-+*/()]+[)\\d]");
boolean match = equationPattern.matcher(text).matches();
System.out.println(match);
if (match) {
// Another class calculates
} else {
clearTimer.restart();
outputField.setText("INPUT ERROR"); // This doesn't appear
}
} else if (e.getSource() == clearTimer) {
outputField.setText("");
}
}