1

我对 Java 中的交互过程有疑问。我有线程来读取 STDOUT 和 STDERR 以及一个线程来处理进程的输入。但是在进程终止之前,STDOUT 流中没有可用的数据。然后是一次打印整个输出。

DBG    | Pipe action-STDERR started
DBG    | Pipe action-STDIN started
DBG    | Pipe action-STDOUT started

STDIN  | Try to put some input.
STDIN  | I cannot see any output.
STDIN  | Nevertheless the interaction works.
STDIN  | It works on background.
STDIN  | Let's terminate the process to see the truth.
STDIN  | quit

STDOUT | Enter some text, please: The text is 'Try to put some input.'
STDOUT | Enter some text, please: The text is 'I cannot see any output.'
STDOUT | Enter some text, please: The text is 'Nevertheless the'
STDOUT | Enter some text, please: The text is 'interaction works.'
STDOUT | Enter some text, please: The text is 'It works on background.'
STDOUT | Enter some text, please: The text is 'Let's terminate the process to see the truth.'
STDOUT | Enter some text, please: The text is 'quit'
STDOUT | Bye!
DBG    | Trying to kill thread action-STDOUT
DBG    | Trying to kill thread action-STDERR
DBG    | Trying to kill thread action-STDIN
DBG    | Pipe action-STDERR finished
DBG    | Finished
DBG    | Pipe action-STDIN finished
DBG    | Pipe action-STDOUT finished

以 STDOUT 为前缀的行是进程写入的行。以 STDIN 为前缀的行是我写的行。以 DBG 为前缀的行是被测试的 Java 程序编写的作为调试信息的行。让我们尝试在系统控制台中执行相同的过程。

Enter some text, please: Text
The text is 'Text'
Enter some text, please: quit
The text is 'quit'
Bye!

这种行为完全符合我的期望。我被要求提供一些意见。我这样做并得到答案。

我很惊讶,我在网上找到了几篇帖子,包括 Stackoverflow,但没有任何答案被标记为可接受的解决方案。(例如,从 Java Process 读取 InputStream 时出现问题。)Java 开发人员似乎从未处理过交互式进程的执行。奇怪的是,非交互式进程的输出(如 ping)在进程运行时按顺序出现。没有任何问题。但是当进程正在等待用户输入时,输出会以某种方式被阻塞。

4

1 回答 1

1

遗憾的是,这不是一个错误,而是一个功能。无论是在 Linux 上还是在 Windows 上,管道都是这样实现的。它们被缓冲以提高性能。一个进程产生数据,第二个进程接管它们。无需交互,该模型在 99% 的情况下也适用。

然而,控制台是另一种应用程序。系统控制台是操作系统的一部分,它的工作方式与我们用 Java 编写的控制台有点不同。当然,系统控制台会立即打印所有数据。一切正常。然后从我们的代码执行的相同过程不会打印提示,尽管它应该工作。怎么了?系统控制台和进程之间没有缓冲区。但是相反,我们的进程和我们的 Java 代码之间有一个缓冲区。这就是原因。

该进程不会自动刷新输出。不管进程是用哪种语言编写的。用 Java、Perl 或 C 编写的进程从不打印提示,直到将显式刷新添加到其代码中。如果您可以影响您将与之交互的进程的代码,请更新代码(伪代码如下):

print 'Give me a number '
flush STDOUT
number = get STDIN
print 'You wrote ' + number
flush STDOUT

没有其他解决方案。如果第三方应用程序没有刷新其输出,则您无法在控制台中使用它。你几乎无法逃避缓冲。尽管如此,还是有可能的。有一些库,或者您可以找到一个可以与有问题的进程交互并检查其代码的开源应用程序。

提示:查看Apache Commons Exec。它是一个专门用于进程执行的库。

于 2012-07-09T07:15:47.440 回答