java SwingWorker 对话框未显示

java SwingWorker dialog is not shown

在 Netbeans 中,我创建了一个 GUI 项目,它使用框架前端,我可以添加组件并双击以编辑它们的事件。我指的是 window,它有 "Source"、"Design" 和 "History" 选项卡。

组件和关系如下:

1- 打开文件选择器的按钮。

2- 一个 TextArea 来显示文件选择器的结果。如果用户选择一个文件,它会在TextArea中显示文件名;否则它会写 "canceled by user".

3- 同时,如果用户选择一个文件,我想打开一个 "please wait" 对话框 SwingWorker 并在后台做一些工作。

问题是,当用户选择一个文件时,我没有看到请稍候对话框!! pastebin 提供了 Netbeans 生成的完整代码。部分代码如下所示:

private void OpenSongFileActionPerformed(java.awt.event.ActionEvent evt) {                                              
    // TODO add your handling code here:
    JFileChooser fileChooser = new JFileChooser();
    fileChooser.setAcceptAllFileFilterUsed(false);
    FileNameExtensionFilter filter = new FileNameExtensionFilter("MP3 files", "mp3");
    fileChooser.addChoosableFileFilter(filter);
    fileChooser.setCurrentDirectory(new File(System.getProperty("user.dir")));
    int result = fileChooser.showOpenDialog(this);
    if (result != JFileChooser.APPROVE_OPTION) {
        //ReadInfo.setText("No song has been selected");
        System.out.println("OpenSongFile canceled by user");
        return;
    }
    final JDialog loading = new JDialog(this);
    JPanel p1 = new JPanel(new BorderLayout());
    p1.add(new JLabel("Please wait..."), BorderLayout.CENTER);
    loading.setUndecorated(true);
    loading.getContentPane().add(p1);
    loading.pack();
    loading.setLocationRelativeTo(this);
    loading.setDefaultCloseOperation(JDialog.DO_NOTHING_ON_CLOSE);
    loading.setModal(true);
    SwingWorker<String, Void> worker = new SwingWorker<String, Void>() {
        @Override
        protected String doInBackground() throws InterruptedException {
            for (int i = 0; i < 10000; i++) 
                for (int j = 0; j < 10000; j++) 
                    ;
            return "hello";
        }
        @Override
        protected void done() {
            loading.dispose();
        }
    };
    worker.execute();
    loading.setVisible(true);
    try {
        worker.get();
    } catch (Exception e1) {
        e1.printStackTrace();
    }

    File selectedFile = fileChooser.getSelectedFile();
    ReadInfo.setText("Selected file: " + selectedFile.getAbsolutePath());
}                              

P.S:我用SwingWorker代码解释here.

worker.get(); 将一直阻塞到 SwingWorker returns,这意味着您正在阻塞 EDT,阻止显示对话框。

相反,利用 SwingWorkers PropertyChangeListener 支持和监控 STARTDONE 事件

SwingWorker<String, Void> worker = new SwingWorker<String, Void>() {
    @Override
    protected String doInBackground() throws InterruptedException {
        Thread.sleep(5000);
        return "hello";
    }

    @Override
    protected void done() {
        loading.dispose();
    }
};
worker.addPropertyChangeListener(new PropertyChangeListener() {
    @Override
    public void propertyChange(PropertyChangeEvent evt) {
        System.out.println(evt.getPropertyName());
        Object value = evt.getNewValue();
        if (value instanceof SwingWorker.StateValue) {
            SwingWorker.StateValue state = (SwingWorker.StateValue) value;
            switch (state) {
                case DONE: {
                    try {
                        String result = worker.get();
                        JOptionPane.showMessageDialog(null, result);
                    } catch (InterruptedException | ExecutionException ex) {
                        ex.printStackTrace();
                    }
                }
                break;
            }
        }
    }
});
worker.execute();
loading.setVisible(true);

另一个问题是您的循环可能 运行 太快以至于 window 在它出现在屏幕上之前就被关闭了。

比如我简单的用Thread.sleep暂停了doInBackground方法5秒...

import java.awt.BorderLayout;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.util.concurrent.ExecutionException;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.swing.JDialog;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import javax.swing.SwingUtilities;
import javax.swing.SwingWorker;

public class Memory extends JFrame {

    public static void main(String[] args) {
        SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                makeItSo();
            }
        });
    }

    public static void makeItSo() {
        final JDialog loading = new JDialog();
        JPanel p1 = new JPanel(new BorderLayout());
        p1.add(new JLabel("Please wait..."), BorderLayout.CENTER);
        loading.setUndecorated(true);
        loading.getContentPane().add(p1);
        loading.pack();
        loading.setLocationRelativeTo(null);
        loading.setDefaultCloseOperation(JDialog.DO_NOTHING_ON_CLOSE);
        loading.setModal(true);
        SwingWorker<String, Void> worker = new SwingWorker<String, Void>() {
            @Override
            protected String doInBackground() throws InterruptedException {
                Thread.sleep(5000);
                return "hello";
            }

            @Override
            protected void done() {
                loading.dispose();
            }
        };
        worker.addPropertyChangeListener(new PropertyChangeListener() {
            @Override
            public void propertyChange(PropertyChangeEvent evt) {
                System.out.println(evt.getPropertyName());
                Object value = evt.getNewValue();
                if (value instanceof SwingWorker.StateValue) {
                    SwingWorker.StateValue state = (SwingWorker.StateValue) value;
                    switch (state) {
                        case DONE: {
                            try {
                                String result = worker.get();
                                JOptionPane.showMessageDialog(null, result);
                            } catch (InterruptedException | ExecutionException ex) {
                                ex.printStackTrace();
                            }
                        }
                        break;
                    }
                }
            }
        });
        worker.execute();
        loading.setVisible(true);
    }

}