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

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

我这样做的方式受到了诸如从java Runtime.exec读取流的建议的启发,并且在function上,工作正常。

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,使用POSIX vfork和https://github.com/axiak/java_posix_spawn这样的库,使用字节缓冲区而不是BufferedReader ……一切都没有正面结果。

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

感谢任何建议!

最诚挚的问候和提前谢谢。

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

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

如果这没有帮助,我认为没有什么可以做,因为其他所有方法都会为您的运行时环境添加更多必须处理的代码。

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