2

我在用 Java 运行外部应用程序时遇到问题。我知道输入和输出流的问题,我正在尝试按如下方式阅读它们:

    ProcessBuilder pb = new ProcessBuilder(args);
    args[args.length - 1] += "<NUL";

    pb.redirectErrorStream(true);
    Process p = pb.start();

    InputStreamReader isr = new InputStreamReader(p.getInputStream());
    BufferedReader input = new BufferedReader(isr);
    while (input.readLine() != null) {
    }       

    p.waitFor();
    input.close();
    isr.close();

我正在执行的命令是 xdg-open 的 URL,所以它不应该等待输入。

这在我的机器上工作十分之九,但有时它只是挂在 pb.start(); 并且该命令未执行。

java 线程正在等待进程返回。这是堆栈跟踪:

Name: LinkHandlerExec
State: WAITING on java.lang.UNIXProcess$Gate@356122dc
Total blocked: 0  Total waited: 1

Stack trace: 
java.lang.Object.wait(Native Method)
java.lang.Object.wait(Object.java:502)
java.lang.UNIXProcess$Gate.waitForExit(UNIXProcess.java:80)
java.lang.UNIXProcess.<init>(UNIXProcess.java:161)
java.lang.ProcessImpl.start(ProcessImpl.java:81)
java.lang.ProcessBuilder.start(ProcessBuilder.java:468)

这是 strace 告诉我的:

$ sudo strace -p 13255
Process 13255 attached - interrupt to quit
futex(0x7fbb673e49d0, FUTEX_WAIT, 13262, NULL
4

1 回答 1

0

我们还在 CentOS 6.3 JDK 1.6.0_33-b03 上观察到了这样的问题。调查表明,这是在该 Java 进程中发生 OutOfMemoryError 之后发生的。

查看类 java.lang.UNIXProcess 的代码,我想当在 UNIXProcess 构造函数中运行的单独线程中抛出非 IO 异常(例如错误)时,可能会发生此类问题,该线程位于第 109 行和第 130 行之间。此代码最好组织为

try {
  .....
} finally {
  gate.exit();
}

, 但它被组织为

try {
  ...
} catch (IOException e) {
  gate.setException(e); /*remember to rethrow later*/
  gate.exit();
  return;
}
...
gate.exit(); 

,因此非 IO 异常可能导致 gate.exit() 不被调用,这导致主调用者线程在方法 gate.waitForExit() 中永远挂起(第 145 行)。

于 2013-05-23T16:29:58.397 回答