我有一个包含 JLabel 的模式对话框。此对话框调用另一个类中的方法,该方法执行长时间运行的查找活动。我希望这个查找类更新从模态对话框传递的 JLabel。
我尝试从查找过程中的代码循环和 Timer 中更新 JLabel。(我发现当模式对话框运行时,Swing 计时器的 actionPerformed 方法永远不会被调用。)自从实现了 Timer,我将查找过程移到了一个单独的线程中。
我的查找有效,但直到过程结束,JLabel 仍然没有重新绘制。我将不胜感激任何帮助。
这是相关的方法。DictionaryTuple 是一个 POJO,包含两个字段:一个 List 和一个 String。只是 getter 和 setter,但 dictionaryThread 确实可以访问 tupleList;请注意,在 dictionaryThread 完成之前,我们不会在这里触摸它。
public List<DictionaryTuple> findTuples() {
tupleList = new ArrayList<DictionaryTuple>();
Thread dictionaryThread = new Thread(this); // This is the long-running lookup thread
dictionaryThread.start();
progressTimer = new Timer();
progressTimer.scheduleAtFixedRate(new TimerTask() {
@Override
public void run() {
// progressCaption is updated in dictionaryThread
synchronized (progressCaption) {
lblProgress.setText(progressCaption);
}
lblProgress.repaint();
Thread.yield();
}
}, 250, 250);
try {
dictionaryThread.join();
} catch (InterruptedException e) {
} finally {
progressTimer.cancel();
lblProgress.setText("");
progressCaption = "";
}
return tupleList;
}
我也尝试过这种变化;在这种情况下,内部的 run() 调用将一直保持到处理完成:
progressTimer.scheduleAtFixedRate(new TimerTask() {
@Override
public void run() {
synchronized (progressCaption) {
System.out.println("Fire!");
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
System.out.println("Update");
lblProgress.setText(progressCaption);
lblProgress.repaint();
}});
}
}
}, 250, 250);