Отображать неопределенный JProgressBar во время выполнения пакетного файла

Я уже некоторое время просматриваю SO и Google, чтобы найти ответ на этот вопрос, но я не могу найти тот, который действительно работает. Начну с самого начала:

Я создал класс Java с методом, который запускает пакетный файл в фоновом режиме (командное окно не отображается). Программа прекрасно работает, за исключением того, что это будет немного смущать конечного пользователя, поскольку пакетный файл занимает некоторое время - пользователь не будет знать, работает программа или нет. После завершения выполнения пакетного сценария появляется диалоговое окно с сообщением о его завершении, но в течение промежутка времени между запуском метода и появлением диалогового окна выглядит так, как будто программа ничего не делает.

Итак, вот мой вопрос: я бы очень хотел отобразить новый фрейм с текстовой областью, которая показывает вывод командного файла. Однако я понимаю, что это очень трудно сделать без создания временных файлов, записи в них, чтения из них и так далее. Я бы предпочел избежать этого, если это возможно. Поэтому я решил, что может быть лучше отобразить неопределенный JProgressBar во время выполнения процесса и закрыть его, когда процесс будет завершен. К сожалению, я не думаю, что Swing может справиться с этим, так как это потребует одновременного запуска нескольких процессов. Я слышал о SwingWorker, но не совсем уверен, как это будет использоваться в этом случае. У меня есть следующий SSCCE, который работает, но не имеет индикатор выполнения.

public myClass(){
    public static void main(String[] args){
        String[] commands = {"cmd.exe", "/C", "C:\\users\\....\\myBat.bat"};
        Process p = Runtime.getRuntime().exec(commands);
        p.waitFor()
        JOptionPane.showMessageDialog(null, "Process finished!");
    }
}

Пока p.waitFor() ожидает процесса, на экране ничего нет. Я просто хочу что-то, показывающее пользователю, что процесс все еще выполняется. Мысли? Спасибо!

1 ответ

Решение

Вы можете запустить ProcessBuilder на фоне SwingWorker, как показано ниже, чтобы получить как вывод, так и индикатор выполнения.

образ

import java.awt.BorderLayout;
import java.awt.EventQueue;
import java.awt.event.*;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import javax.swing.*;

/**
 * @se http://stackru.com/a/20603012/230513
 * @see http://stackru.com/a/17763395/230513
 */
public class SwingWorkerExample {

    private final JLabel statusLabel = new JLabel("Status: ", JLabel.CENTER);
    private final JTextArea textArea = new JTextArea(20, 20);
    private JButton startButton = new JButton("Start");
    private JButton stopButton = new JButton("Stop");
    private JProgressBar bar = new JProgressBar();
    private BackgroundTask backgroundTask;
    private final ActionListener buttonActions = new ActionListener() {
        @Override
        public void actionPerformed(ActionEvent ae) {
            JButton source = (JButton) ae.getSource();
            if (source == startButton) {
                textArea.setText(null);
                startButton.setEnabled(false);
                stopButton.setEnabled(true);
                backgroundTask = new BackgroundTask();
                backgroundTask.execute();
                bar.setIndeterminate(true);
            } else if (source == stopButton) {
                backgroundTask.cancel(true);
                backgroundTask.done();
            }
        }
    };

    private void displayGUI() {
        JFrame frame = new JFrame("Swing Worker Example");
        frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);

        JPanel panel = new JPanel();
        panel.setBorder(
            BorderFactory.createEmptyBorder(5, 5, 5, 5));
        panel.setLayout(new BorderLayout(5, 5));

        JScrollPane sp = new JScrollPane();
        sp.setBorder(BorderFactory.createTitledBorder("Output: "));
        sp.setViewportView(textArea);

        startButton.addActionListener(buttonActions);
        stopButton.setEnabled(false);
        stopButton.addActionListener(buttonActions);
        JPanel buttonPanel = new JPanel();
        buttonPanel.add(startButton);
        buttonPanel.add(stopButton);
        buttonPanel.add(bar);

        panel.add(statusLabel, BorderLayout.PAGE_START);
        panel.add(sp, BorderLayout.CENTER);
        panel.add(buttonPanel, BorderLayout.PAGE_END);

        frame.setContentPane(panel);
        frame.pack();
        frame.setLocationByPlatform(true);
        frame.setVisible(true);
    }

    private class BackgroundTask extends SwingWorker<Integer, String> {

        private int status;

        public BackgroundTask() {
            statusLabel.setText((this.getState()).toString());
        }

        @Override
        protected Integer doInBackground() {
            try {
                ProcessBuilder pb = new ProcessBuilder("ls", "-lR", "/");
                pb.redirectErrorStream(true);
                Process p = pb.start();
                String s;
                BufferedReader stdout = new BufferedReader(
                    new InputStreamReader(p.getInputStream()));
                while ((s = stdout.readLine()) != null && !isCancelled()) {
                    publish(s);
                }
                if (!isCancelled()) {
                    status = p.waitFor();
                }
                p.getInputStream().close();
                p.getOutputStream().close();
                p.getErrorStream().close();
                p.destroy();
            } catch (IOException | InterruptedException ex) {
                ex.printStackTrace(System.err);
            }
            return status;
        }

        @Override
        protected void process(java.util.List<String> messages) {
            statusLabel.setText((this.getState()).toString());
            for (String message : messages) {
                textArea.append(message + "\n");
            }
        }

        @Override
        protected void done() {
            statusLabel.setText((this.getState()).toString() + " " + status);
            stopButton.setEnabled(false);
            startButton.setEnabled(true);
            bar.setIndeterminate(false);
        }

    }

    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                new SwingWorkerExample().displayGUI();
            }
        });
    }
}
Другие вопросы по тегам