进度监视器(Indeterminate)显示执行发生在后端

时间:2013-04-09 12:16:39

标签: java multithreading process progress-bar processbuilder

我在我的java程序中使用Processbuilder在后端执行外部进程。我正在使用process.waitFor来获取进程的退出状态。后端进程需要花费时间,这取决于所提供的输入。因此,我需要在进程完成之前创建一个不确定时间的进度条。这只是为了通知用户后端正在进行处理。

    try {
        button.setEnabled(false);
        this.progressbar.setVisible(true);
        this.progressbar.setIndeterminate(true);
        }
        catch(Exception e)
        {
            e.printStackTrace();
        }
        perform(this);
    }
}                                    

public void perform(JFrame f)
{
            String cmd[]={"cmd", "/c","a.exe:};
            ProcessBuilder pb = new ProcessBuilder(cmd);
            pb.directory(new File(dir_path));
            pb.redirectErrorStream(true);
            try {
                    p = pb.start();
                } catch (IOException ex) {
                    Logger.getLogger(execution.class.getName()).log(Level.SEVERE, null, ex);
                       }
            try
            {
               int exitVal = p.waitFor();
               if(exitVal>0)
               {
                   this.progressbar.setVisible(false);
                   this.progressbar.setIndeterminate(false);
                   this.button.setEnabled(true);
               }
               if(p.waitFor()==0)
               {
                   this.progressbar.setVisible(false);
                   this.progressbar.setIndeterminate(false);

                   JOptionPane.showMessageDialog(f,"Scan completed successfully.");
               }
             } catch (InterruptedException ex) {
    Logger.getLogger(execution.class.getName()).log(Level.SEVERE, null, ex);
}
}

按钮是调用该调用的按钮的名称。但我面临的问题是函数调用执行是在不执行这些语句的情况下完成的。

    try {
        button.setEnabled(false);
        this.progressbar.setVisible(true);
        this.progressbar.setIndeterminate(true);
        }
        catch(Exception e)
        {
            e.printStackTrace();
        }

执行该功能后禁用按钮。 1)我可以知道为什么会这样吗? 2)我可以知道使进度条指示后端进度直到完成的解决方案吗?

提前致谢。

2 个答案:

答案 0 :(得分:2)

我建议使用SwingWorker。这是为了解决这个问题。以下是我的代码库中的一些代码的代码段(vui是一个JFrame):

vui.clearOutput();
vui.setOutput("Waiting for items to copy...\n"
            + "This could take several minutes. Please standby...");
vui.disableExit();
vui.disableInput();
vui.disableNext();
vui.showProgressBar();

// make thread so you can disable all options when zipping and enable progress bar
SwingWorker transferWorker = new SwingWorker() {
    @Override
    protected Void doInBackground() throws Exception {
        try {
            Process p = Runtime.getRuntime().exec("resources/bin/transferCopier.bat");

            StreamGobbler errorGobbler = new StreamGobbler(p.getErrorStream(), "ERROR");
            StreamGobbler outputGobbler = new StreamGobbler(p.getInputStream(), "OUTPUT");
            errorGobbler.start();
            outputGobbler.start();

            p.waitFor();
        } catch (IOException ex) {
            Logger.getLogger(VaderController.class.getName()).log(Level.SEVERE, null, ex);
        } catch (InterruptedException ie) {
            Logger.getLogger(VaderController.class.getName()).log(Level.SEVERE, null, ie);
        }
        return null;
    }
    @Override
    public void done() {
        vui.hideProgressBar();
        vui.clearOutput();
        vui.setOutput("Transfer Complete!\n\n"
                    + "Push \"Exit\" to quit.\n\n");
        vui.enableExit();
        mode = VaderMode.END;
    }
};
transferWorker.execute();

执行transferWorker.execute();时,会调用doInBackground()。完成doInBackground()计算后,将调用done()done()是您对GUI进行任何最终更新的地方。

关于我上面代码的关键注意事项是我在执行SwingWorker之前启用了进度条,然后当SwingWorker执行完毕时,我禁用了进度条。

资源:

http://docs.oracle.com/javase/6/docs/api/javax/swing/SwingWorker.html http://docs.oracle.com/javase/tutorial/uiswing/concurrency/worker.html

答案 1 :(得分:0)

它被执行,但它在与GUI相同的线程中执行,因此视图不会更新。

显示ProgressBar并禁用Button后,启动一个等待Process完成的runnable。

只需将perform()包裹在Runnable中。

未经测试的代码: 而不是perform(this)你写的:

new Thread(new Runnable() { 
    public void run() {
        perform(YourClassName.this);
    }
}).start();