5

我的问题是一个基于理论的问题,但它确实满足了我的特定需求。

当您的 SwingWorker 抛出一个您 a) 可以预期并且 b) 需要从中恢复并继续的异常时,您会怎么做,但您想通知用户此错误已发生?如何在不违反“No Swing code from doInBackground()”规则的情况下获取预期的异常并通知用户?

考虑到这个问题,我开发了一个 SSCCE,我想提出下面的问题。

SSCCE:

import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.concurrent.ExecutionException;
import javax.swing.JButton;
import javax.swing.JDialog;
import javax.swing.JFrame;
import javax.swing.JOptionPane;
import javax.swing.JProgressBar;
import javax.swing.SwingUtilities;
import javax.swing.SwingWorker;

public class Test {

    public static void main(String args[]) {
        final JFrame frame = new JFrame();
        JButton go = new JButton("Go.");
        go.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent e) {
                new Task(frame);
            }
        });
        frame.add(go);
        frame.pack();
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frame.setLocationByPlatform(true);
        frame.setVisible(true);
    }

    static class Task extends SwingWorker<Void, Void> {
        JFrame parent;
        JDialog dialog;
        public Task(JFrame parent) {
            this.parent = parent;
            dialog = new JDialog(parent);
            JProgressBar jpb = new JProgressBar();
            jpb.setIndeterminate(true);
            dialog.add(jpb);
            dialog.setLocationRelativeTo(null);
            dialog.setVisible(true);
            execute();
        }

        @Override
        protected Void doInBackground() throws Exception {
            for(int i = 0; i < 100000; i++) {
                System.out.println(i);
                try {
                    if(i == 68456) throw new IllegalStateException("Yikes! i = 68456.");
                } catch (final IllegalStateException e) {
                    SwingUtilities.invokeAndWait(new Runnable() {
                        @Override
                        public void run() {
                            JOptionPane.showMessageDialog(parent, "Error: " + e.getMessage(), "Error", JOptionPane.ERROR_MESSAGE);
                        }
                    });
                }
            }
            return null;
        }

        @Override
        protected void done() {
            if (!isCancelled()) {
                try {
                    get();
                } catch (ExecutionException | InterruptedException e) {
                    e.printStackTrace();
                }
            }
            System.out.println("done");
            dialog.dispose();
        }

    }
}

SwingUtilities.invokeAndWait()在方法内调用是否有效doInBackground()我对此进行了一些线程分析,结果如下:

在此处输入图像描述

按下“开始”按钮后,SwingWorker-pool-1-thread-1线程变为绿色。然后,当满足 if 条件时,抛出错误,并显示错误对话框,线程变为黄色,线程上确实有一个绿色的“blip” AWT-EventQueue-0,表示调用了 EDT。我等了大约 10 秒钟,按下“确定”,SwingWorker线程又变绿了。

做这样的事情有什么潜在的陷阱吗?有没有人有实时通知用户计算错误的经验SwingWorker

老实说,这种方法让我很怀疑。这似乎很不正统,但我不能肯定这是否是一个坏主意。

4

1 回答 1

4

invokeAndWait()当用户实际需要批准时,我认为使用没有问题。如果没有,如本所示,aSwingWorker<Void, String>可以简单地调用publish()交错的数据和错误消息。如有必要,附加的适当消息done()将允许用户查看累积的输出。

于 2013-12-14T01:08:51.547 回答