在伪代码中,这就是我正在做的事情:
Process proc = runtime.exec(command);
processOutputStreamInThread(proc.getInputStream());
processOutputStreamInThread(proc.getErrorStream());
proc.waitFor()
但是,有时processOutputStreamInThread
看不到任何输出,有时会看到。粗略地说,该方法创建BufferedInputStream
命令的输出并将其发送到记录器。
根据我所看到的,我猜测command
不需要将所有输出转储到由getInputStream()
and馈送的流getErrorStream()
中,从而允许流为空。
我的试验结果是以下问题:
(1) java.lang.ProcesswaitFor()
中是否要求执行程序的输出在返回之前已被读取?
该文档仅说明:
如有必要,使当前线程等待,直到此
Process
对象表示的进程终止。如果子进程已经终止,则此方法立即返回。如果子进程尚未终止,则调用线程将被阻塞,直到子进程退出。
(2)在什么条件下,由提供的流需要关闭和/或自动关闭getInputStream
?getErrorStream
该文档仅说明:
获取子流程的错误流。该流从该 Process 对象表示的进程的错误输出流中获取通过管道传输的数据。
实施说明:缓冲输入流是个好主意。
一位用户报告说他必须自己关闭流,但至少在部分时间我得到一个异常,表明当我尝试这样做时流已经关闭。
编辑:更改getOutputStream
为getInputStream
,现在出现在上面。
解决方案:问题最终是在某些情况下,用于处理输出流的线程直到我的非常短暂的进程完成后才会运行,导致输入流没有给我任何数据。 waitFor
没有做任何等待执行程序的输出。相反,程序在收集任何输出之前运行并终止。
我使用线程是因为我不确定我将在标准错误和标准输出上获得多少输出,并且我希望能够同时处理这两者,而不会阻塞其中一个或另一个应该只有其中一个有可用数据。但是,因为我的线程不能一致地读取执行程序的输出,所以这是一个非解决方案。
我的最终编码看起来像这样:
ProcessBuilder pb = new ProcessBuilder(cmdargs);
pb.redirectErrorStream(true);
Process proc = pb.start();
processOutputStream(proc.getInputStream());
proc.waitFor()