使用Runtime.exec()生成的进程的高效执行和输出流重定向

时间:2013-10-29 10:31:33

标签: java multithreading performance stream runtime.exec

我有一个脚本可以执行多次程序,为STDERR和STDOUT生成大约350行输出。现在,我需要在Java中执行脚本,从而将输出流打印到其原始目标。所以,基本上,我从Java类中执行脚本,维护用户的原始行为。

我这样做的方式受到Reading streams from java Runtime.exec之类的建议的启发,功能上也很好。

Process p = Runtime.getRuntime().exec(cmdarray);
new Thread(new ProcessInputStreamHandler(p.getInputStream(), System.out)).start();
new Thread(new ProcessInputStreamHandler(p.getErrorStream(), System.err)).start();
return p.waitFor();

班级ProcessInputStreamHandler

class ProcessInputStreamHandler implements Runnable {
    private BufferedReader in_reader;
    private PrintStream out_stream;

    public ProcessInputStreamHandler(final InputStream in_stream, final PrintStream out_stream) {
        this.in_reader  = new BufferedReader(new InputStreamReader(in_stream));
        this.out_stream = out_stream;
    }

    @Override public void run() {
        String line;
        try {
            while ((line = in_reader.readLine()) != null) {
                out_stream.println(line);
            }
        } catch (Exception e) {throw new Error(e);}

        out_stream.flush();
    }
}

现在关于我的问题陈述:虽然脚本的执行大约需要17秒,但“封装”执行至少需要21秒。我在哪里丢失这些4秒或更长时间?

我已经尝试使用ProcessBuilder将STDERR重定向到STDOUT,使用带有https://github.com/axiak/java_posix_spawn等库的POSIX vfork,使用字节缓冲区而不是BufferedReader ...所有内容都没有结果是积极的。

有什么建议吗?我知道会有一些性能损失,但4秒对我来说似乎有点太多了......

感谢任何建议!

最好的问候和提前谢谢。

2 个答案:

答案 0 :(得分:2)

您的任务的最快方法是使用Java 7和

return new ProcessBuilder(cmdarray).inheritIO().start().waitFor();

如果这没有帮助,我认为你无能为力,因为其他每种方法都会为你的运行时环境添加更多必须处理的代码。

答案 1 :(得分:1)

不知道它是否会提高性能,但您可以尝试使用NuProcess库,同时提供非阻塞(异步)I / O也将在Linux上使用vfork,这确实会减少进程启动时间(和内存开销)相当多。