import javaio*;
public class CommandWrapper
{
Process process;
Thread in;
Thread out;
public CommandWrapper(Process process)
{
thisprocess = process;
final InputStream inputStream =
processgetInputStream();
//final BufferedReader
r=new BufferedReader
(new InputStreamReader(inputStream));
final byte[] buffer = new byte[];
out = new Thread()
{
//String line;
int lineNumber=;
public void run()
{
try {
while (true)
{
int count = inputStreamread(buffer);
Systemoutprintln
(lineNumber+:+new String
(buffer count));
//line=rreadLine();
//Systemoutprintln
(lineNumber+:+line);
lineNumber++;
}
}
catch (Exception e)
{
}
}
};
final BufferedReader reader =
new BufferedReader
(new InputStreamReader(Systemin));
final OutputStream
outputStream = processgetOutputStream();
in = new Thread()
{
String line;
public void run()
{
try
{
while (true)
{
outputStreamwrite(
(readerreadLine()+\n)getBytes());
outputStreamflush();
}
}
catch (Exception e)
{
}
}
};
}
public void startIn()
{
instart();
}
public void startOut()
{
outstart();
}
public void interruptIn()
{
ininterrupt();
}
public void interruptOut()
{
outinterrupt();
}
public static void main
(String[] args)
{
try
{
CommandWrapper command
= new CommandWrapper
(RuntimegetRuntime()exec(nativeascii));
commandstartIn();
commandstartOut();
}
catch (Exception e)
{
eprintStackTrace();
}
}
}
我以nativeascii为范例程序和网友给我的那个程序做了对比发现如下几个在处理这个问题时需要注意的地方
由于不知道目标程序的输入输出顺序因此只能建立两个单独的线程分别处理输入和输出这样输入和输出就不会阻塞了但是有些目标程序要求有特定的输入输出顺序而经过这个类封装的结果是在任何状态下都可以输入程序的任何输出也会被马上反映出来这是构造通用类的第一个问题
不能直接使用I/O重定向在最开始的时候我是考虑直接使用I/O重定向的但是实际的情况是Process的I/O的定义刚好和我的预想相反我们从Process取得的InputStream实际上是它的输出而取得的OutputStream是它的输入这样就无法进行I/O重定向了必须我们进行编码来读取程序的输出和写入控制台的输入(这里的I/O重定向是指想将它的I/O直接重定向到系统的I/O)
写入控制台的输入
outputStreamwrite(
(readerreadLine()+\n)getBytes());
outputStreamflush();
这里有两个问题值得注意第一个是我们在控制台输入一行数据以后按下回车那么语句readerreadLine()可以正确的得到你的输入为什么要加那个换行符呢?这是在测试的时候发现的问题在以nativeascii作为例子的时候发现不加这个的话它不能得到控制台的输入但是我在替那位网友解决的问题的时候他的程序则没有这个问题
因此猜想可能是因为有的程序要求读取的一整行的数据(例如nativeascii)而大部分的命令行程序在编码的时候读取的是整数这样的值或者其他类型的值他们是以空格或者其他的字符分隔的因此就不需要那个额外的换行符(例如那位网友的程序读取的是一元二次方程的三个系数)
另外一个问题就是flush方法的使用在最开始的时候没有想到要这样刷新进去无论是否加换行符外部程序都无法读取写入的输入后来才想到要调用一下这个方法这个也是在我们输出的时候应该注意的一个问题有些需要马上反应出来的输出一般都在写入以后要调用它否则输出/输入不能马上反应出来
对于程序的输出最开始我是构造的一个BufferedReader想以行为单位输出对于那位网友的程序结果证明不是很好用但是以nativeascii作为例子运行又没有问题这个估计和外部程序的代码也有关系如果外部程序没有输出换行符可能使用BufferedReader就会有问题但是通过直接读取输出就没有问题了另外需要注意的就是
Systemoutprintln
(lineNumber+:+new String
(buffer count));
中严格来说应该是
Systemoutprintln
(lineNumber+:+new String
(buffer count));
之所以减一是因为读取输入的时候人为的多加了一个换行符如果这个地方不减一就会多输出一个空行
基于以上的种种原因要构造一个执行外部程序的包装器类不太好办特别是文章中提到的几个问题有时间和兴趣的朋友可以做一下测试看看以上的问题和猜测是否正确
另外附上一段源代码是一个fortran的程序
implicit none
real abc
real d
real rootroot
print*请输入一元二次方程
的系数abc:
read(**) abc
d=b***a*c
if(d>=) then
root=(b+sqrt(d))/(*a)
root=(bsqrt(d))/(*a)
print*root=root
print*root=root
else
print*一元二次方程没有实根!
end if
pause
end