我是否在EDT之外更新Swing组件?

Yes*_*MPS 8 java swing multithreading swingworker event-dispatch-thread

我一直在阅读很多关于Swing,线程,invokeLater(),SwingWorker等的内容,但我似乎无法理解这一切,所以我试图创建一个非常简单的程序来说明.我看了很多例子,但似乎没有一个例子表明我正在尝试做什么.

这是我在我的例子中想要做的事情.我有一个按钮和一个标签,当我单击按钮时,我希望程序暂停3秒钟,然后在标签文本中添加句点.在这3秒钟内,我希望GUI正常显示并继续响应额外的点击.这是我写的:

import javax.swing.SwingWorker;

public class NewJFrame extends javax.swing.JFrame
{
    private javax.swing.JButton jButton1;
    private javax.swing.JLabel jLabel1;

    public NewJFrame()
    {
        initComponents();
    }
    private void initComponents()
    {
        jButton1 = new javax.swing.JButton();
        jLabel1 = new javax.swing.JLabel();
        setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
        jButton1.setText("Button");
        jButton1.addActionListener(new java.awt.event.ActionListener()
        {
            public void actionPerformed(java.awt.event.ActionEvent evt)
            {
                jButton1ActionPerformed(evt);
            }
        });
        getContentPane().add(jButton1, java.awt.BorderLayout.CENTER);
        jLabel1.setText("Text");
        getContentPane().add(jLabel1, java.awt.BorderLayout.PAGE_END);
        pack();
    }

    private void jButton1ActionPerformed(java.awt.event.ActionEvent evt)
    {
        SwingWorker worker=new SwingWorker()
        {
            protected Object doInBackground()
            {
                try{Thread.sleep(3000);}
                catch (InterruptedException ex){}
                return null;
            }
        };
        worker.execute();
        jLabel1.setText(jLabel1.getText()+".");
    }

    public static void main(String args[])
    {
        java.awt.EventQueue.invokeLater(new Runnable()
        {
            public void run(){ new NewJFrame().setVisible(true); }
        });
    }
}
Run Code Online (Sandbox Code Playgroud)

使用此代码,如果我单击按钮,则句点会立即附加到标签上,这对我来说很有意义,因为我正在创建并睡眠后台线程,让EDT可以立即更新标签.所以我尝试了这个:

private void jButton1ActionPerformed(java.awt.event.ActionEvent evt)
{
    Thread thread = new Thread();
    try{ thread.sleep(3000);}
    catch (InterruptedException ex){}
    jLabel1.setText(jLabel1.getText()+".");
}
Run Code Online (Sandbox Code Playgroud)

这几乎可以工作,除了它阻止EDT导致按钮变为蓝色三秒钟,然后将句点附加到标签的文本.我不希望按钮看起来像是在整个三秒钟被按下时真的只是快速点击,所以我尝试了这个:

private void jButton1ActionPerformed(java.awt.event.ActionEvent evt)
{
    SwingWorker worker=new SwingWorker()
    {
        protected Object doInBackground()
        {
            try{Thread.sleep(3000);}
            catch (InterruptedException ex){}
            jLabel1.setText(jLabel1.getText()+".");
            return null;
        }
    };
    worker.execute();
}
Run Code Online (Sandbox Code Playgroud)

这似乎工作,但我不是从后台线程而不是EDT调用jLabel1.setText(...),因此打破了"Swing单线程规则?" 如果是这样,有没有更好的方法来达到预期的效果?如果没有,你能解释一下原因吗?

Mad*_*mer 4

你真的很接近...

尝试这样的事情。

SwingWorker worker=new SwingWorker()
{
    protected Object doInBackground()
    {
        try{
            Thread.sleep(3000);
        }catch (InterruptedException ex){}
        return null;
    }

    // This is executed within the context of the EDT AFTER the worker has completed...    
    public void done() {
        jLabel1.setText(jLabel1.getText()+".");
    }
};
worker.execute();
Run Code Online (Sandbox Code Playgroud)

您可以使用以下命令检查您是否在 EDT 中运行EventQueue.isDispatchingThread()

更新

您也可以使用javax.swing.Timer可能更容易的...

Timer timer = new Timer(3000, new ActionListener() {
    public void actionPerformed(ActionEvent evt) {
        jLabel1.setText(jLabel1.getText()+".");
    }
});
timer.setRepeats(false);
timer.start();
Run Code Online (Sandbox Code Playgroud)