2

我正在编写一个示例来展示 SwingUtilities.invokeLater() 的用法。我开始意识到我编写的这个简单示例(包括代码)并不保证使用 invokeLater()。我确实遇到过需要使用 invokeLater() 的时候,但我忘记了使用的地点和时间。我也明白在非 EDP 线程中我应该使用 invokeLater() 但在我的情况下我似乎不需要它并且它工作正常。我希望有人告诉我为什么我不需要在这段代码中使用 invokeLater() 。我希望我的展示代码中没有错误。

顺便说一句,我在 Linux 和 Windows 中使用 JDK 1.6 / 1.7。

谢谢。

import javax.swing.*;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

public class HelloButton {
    public static void main(String[] args) {
        JFrame f = new JFrame();
        f.setLayout(new FlowLayout());
        f.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
        JSplitPane pane = new JSplitPane();
        f.add(pane);
        final JLabel clickMessage = new JLabel("No Message at " + System.currentTimeMillis());
        pane.setLeftComponent(clickMessage);
        JButton clickMe = new JButton("Click me");
        clickMe.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent e) {
                threadedIntensiveProcess(clickMessage);
//                intensiveProcess(clickMessage);
            }
        });
        pane.setRightComponent(clickMe);

        f.pack();
        f.setVisible(true);
    }

    static private void threadedIntensiveProcess(final JLabel label)
    {
        new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("UI Thread : " + SwingUtilities.isEventDispatchThread());
                intensiveProcess(label);
            }
        }).start();
    }

    static private void intensiveProcess(JLabel label)
    {
        label.setText("was clicked at " + System.currentTimeMillis());
        for (int i = 0; i < 3; i++)
        {
            try {
                Thread.sleep(2000);
            } catch (InterruptedException e1) {
                e1.printStackTrace();
            }
            label.setText("was updated at " + System.currentTimeMillis());
        }
        System.out.println(label.getText());
    }
}
4

1 回答 1

3

这里有两种情况您应该使用invokeLater.

  1. 用户界面的创建。
  2. 的更新JLabel

UI 的整个创建应该在 EDT 中完成。在古老的 Swing 时代,据说您可以在主线程之前showsetVisible(true)在主线程中进行任何创建,因为在此之前不会出现线程问题。由于这是并且是值得怀疑的,现在不鼓励这样做;它可能会起作用 - 显然,就像你的情况一样 - 但不能保证。

你很幸运,setTextJLabel这种情况下表现良好。由于 API 不保证该方法是线程安全的,因此您应该在 EDT 中调用此方法,就像对 Swing 方法的任何其他调用一样。

所以,总而言之:你有一个简单的例子似乎有效——因为它很简单而且你很幸运。不要依赖这样的“测试”,而是依赖文档。如果您的展示涉及代码的呈现,那么您必须将调用移至 EDT 以确保不会出错。

我不明白为什么你的代码不应该“保证”使用invokeLater.

于 2012-02-21T06:59:36.820 回答