0

我正在尝试学习ProgressMonitorJava Swing。我创建了这个简单的测试代码 -

public class ProgressMonitorTest extends JFrame 
{
private JPanel contentPane;
private ProgressMonitor progressMonitor;
private JButton button;
private static ProgressMonitorTest frame;
private static boolean isFrameReady;

public JButton getButton()
{
    return button;
}

public ProgressMonitor getProgressMonitor()
{
    return progressMonitor;
}

/**
 * Launch the application.
 */
public static void main(String[] args) 
{
    EventQueue.invokeLater(new Runnable() 
    {
        public void run() 
        {
            try 
            {
                frame = new ProgressMonitorTest();
                frame.setVisible(true);
                isFrameReady = true;
            } 
            catch (Exception e) 
            {
                e.printStackTrace();
            }
        }
    });

    while(!isFrameReady)
    {
        //
    }

    frame.getButton().addActionListener(new ActionListener() 
    {   
        @Override
        public void actionPerformed(ActionEvent e) 
        {
            try
            {
                for(int i=0;i<=10;i++)
                {
                    final int percent = i;
                    SwingUtilities.invokeAndWait(new Runnable() 
                    {   
                        @Override
                        public void run()
                        {
                            frame.getProgressMonitor().setProgress(percent * 10);
                            frame.getProgressMonitor().setNote("Completed " + percent*10 + "%.");
                        }
                    });
                    try
                    {
                        Thread.sleep(1000);
                    }
                    catch(Exception ee)
                    {
                        //
                    }                           
                }
            }
            catch(Exception es)
            {
                //
            }
        }           
    });
}

/**
 * Create the frame.
 */
public ProgressMonitorTest() 
{
    isFrameReady = false;

    setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    setBounds(100, 100, 450, 300);
    setTitle("Progress Monitor");

    contentPane = new JPanel();
    contentPane.setBorder(new EmptyBorder(5, 5, 5, 5));
    contentPane.setLayout(new BorderLayout(0, 0));

    progressMonitor = new ProgressMonitor(frame, "Update in progress...", "", 0, 10);
    button = new JButton("Click Here");
    contentPane.add(button);

    setContentPane(contentPane);
}

}

关于这个的几个问题——

  1. 如果我取消isFrameReady选中,程序NullPointerException会在我分配按钮的动作侦听器的那一行显示 a。

  2. 如果我保留上述检查,则单击该按钮不会执行任何操作。

  3. 保持上述检查然后调试它,我让它等待一段时间,然后它到达动作监听器所在的行。在这种情况下,它可以工作,但会立即退出,说它不能invokeAndWait从事件处理线程调用。

我在这一切中缺少什么?有人可以解释如何让它工作。

4

1 回答 1

2

如果我删除 isFrameReady 检查,程序会在我分配按钮的动作侦听器的那一行显示 NullPointerException。

您的使用isFrameReady确保您已frame成功创建。在您的内部main,您使用 call 向事件调度线程(EDT)发布请求EventQueue.invokeLater(new Runnable(){}):删除检查isFrameReady,您将frame.getButton()在主线程中调用,但 frame尚未frame = new ProgressMonitorTest(); 在 EDT 中创建,因此NullPointerException发生了。

如果我保留上述检查,则单击该按钮不会执行任何操作。

你现在应该明白了,上面的检查与按钮点击无关。该按钮没有做任何事情,因为 GUI 因违反 swing 的单线程规则而被冻结。将方法的递增for循环actionPerformed放在另一个线程中,如下面的代码片段所示,并从那里执行它。你会看到它工作正常。

 new Thread(){
    public void run()
    {
       for(int i=0; i<10; i++)
        {
           //whatever you were doing.
        }
    }
   }.start(); 

保持上述检查然后调试它,我让它等待一段时间,然后它到达动作监听器所在的行。在这种情况下,它可以工作,但会立即退出,说它不能从事件处理线程调用 invokeAndWait。

SwingUtitlies.invokeAndWait()阻塞当前线程并等待直到 EDT 完成执行分配给它的任务。由于actionPerformed()函数已经在 EDT 中运行,因此SwingUtitlies.invokeAndWait()从当前线程:EDT 调用会阻塞当前线程:EDT,这是不允许的。不要invokeAndWait用于这种情况。你应该打电话SwingUtilities.invokeLater()

但是,在您了解 Swing 线程模型之前,我认为您不会得到任何东西。阅读 javadoc 和一些互联网资源。拥有《肮脏的富客户》一书,并尝试本书提供的示例:您将在图形效果方面拥有比任何其他资源都提供的更多知识。

于 2013-10-07T17:58:38.730 回答