在方法上使用 synchronized 关键字一次只允许一个线程执行该方法,但 EDT 可以处理将在该方法中同时运行的多个“事件”。请参阅下面的示例代码进行演示。单击测试按钮时,输出为:
0 before dialog, EDT=true
1 before dialog, EDT=true
(click OK button for 1 here)
1 after dialog, EDT=true
(click OK button for 0 here)
0 after dialog, EDT=true
我正在寻找的是一种方法,一次只允许一个 EDT 事件在 test() 方法中处于活动状态,以便输出为
0 before dialog, EDT=true
(click OK button for 0 here)
0 after dialog, EDT=true
1 before dialog, EDT=true
(click OK button for 1 here)
1 after dialog, EDT=true
好像以前一定有人解决过这个问题。我认为可以在方法的开头编写某种锁定对象来使用,或者将方法包装起来,但是懒惰,宁愿不重新发明轮子。
我的测试用例:
package test1;
import javax.swing.JDialog;
import javax.swing.JOptionPane;
import javax.swing.SwingUtilities;
import javax.swing.SwingWorker;
public class EDTSyncTest extends javax.swing.JFrame {
private static final Object locker = new Object();
private int counter;
public EDTSyncTest() {
initComponents();
}
private synchronized void test() {
int l_id = counter++;
logit("" + l_id + " before dialog, EDT=" + SwingUtilities.isEventDispatchThread());
JOptionPane l_pane = new JOptionPane("test id " + l_id);
JDialog l_diag = l_pane.createDialog(this, "test");
l_diag.setModal(true);
l_diag.setVisible(true);
logit("" + l_id + " after dialog, EDT=" + SwingUtilities.isEventDispatchThread());
}
private void startTest() {
new Delayer().execute();
test();
}
private static void logit(String a_msg) {
System.out.println(a_msg);
}
private class Delayer extends SwingWorker<Object, Object> {
@Override
protected Object doInBackground() throws Exception {
Thread.sleep(2000);
return null;
}
@Override
protected void done() {
test();
}
}
private void initComponents() {
jButton1 = new javax.swing.JButton();
setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
getContentPane().setLayout(new java.awt.FlowLayout());
jButton1.setText("Test");
jButton1.setName("jButton1"); // NOI18N
jButton1.addActionListener(new java.awt.event.ActionListener() {
public void actionPerformed(java.awt.event.ActionEvent evt) {
jButton1ActionPerformed(evt);
}
});
getContentPane().add(jButton1);
pack();
}
private void jButton1ActionPerformed(java.awt.event.ActionEvent evt) {
startTest();
}
public static void main(String args[]) {
java.awt.EventQueue.invokeLater(new Runnable() {
public void run() {
new EDTSyncTest().setVisible(true);
}
});
}
protected javax.swing.JButton jButton1;
}