您的位置:首页 > 编程语言 > Java开发

Java中Process和Runtime()使用,以及调用cmd命令阻塞在process.waitfor( )的问题解决

2015-11-05 15:44 961 查看
转自:http://249wangmang.blog.163.com/blog/static/52630765201261334351635/

最近在java中调用perl程序,由于perl中使用斯坦福分词器,有很多控制台输出,导致一直阻塞在process.waitfor( ),只有强制终止java程序后,结果文件才会输出。根据下面两个博客内容成功解决。

用Java编写应用时,有时需要在程序中调用另一个现成的可执行程序或系统命令,这时可以通过组合使用Java提供的Runtime类和Process类的方法实现。下面是一种比较典型的程序模式:

  Process process = Runtime.getRuntime().exec("p.exe");

最近在java中调用perl程序,由于perl中使用斯坦福分词器,有很多控制台输出,导致一直阻塞在process.waitfor( ),只有强制终止java程序后,结果文件才会输出。根据下面两个博客内容成功解决。

用Java编写应用时,有时需要在程序中调用另一个现成的可执行程序或系统命令,这时可以通过组合使用Java提供的Runtime类和Process类的方法实现。下面是一种比较典型的程序模式:

  Process process = Runtime.getRuntime().exec("p.exe");

  process.waitfor( );

在上面的程序中,第一行的“p.exe”是要执行的程序名;Runtime.getRuntime()返回当前应用程序的Runtime对象,该对象的exec()方法指示Java虚拟机创建一个子进程执行指定的可执行程序,并返回与该子进程对应的Process对象实例。通过Process可以控制该子进程的执行或获取该子进程的信息。第二条语句的目的等待子进程完成再往下执行。但在windows平台上,如果处理不当,有时并不能得到预期的结果。下面是笔者在实际编程中总结的几种需要注意的情况:

  1、执行DOS的内部命令如果要执行一条DOS内部命令,有两种方法。一种方法是把命令解释器包含在exec()的参数中。例如,执行dir命令,在NT上,可写成exec ("cmd.exe /c dir"),在windows 95/98下,可写成“command.exe/c dir”,其中参数“/c”表示命令执行后关闭Dos立即关闭窗口。另一种方法是,把内部命令放在一个批命令my_dir.bat文件中,在Java程序中写成exec("my_dir.bat")。如果仅仅写成exec("dir"),Java虚拟机则会报运行时错误。前一种方法要保证程序的可移植性,需要在程序中读取运行的操作系统平台,以调用不同的命令解释器。后一种方法则不需要做更多的处理。

   2、打开一个不可执行的文件打开一个不可执行的文件,但该文件存在关联的应用程序,则可以有两种方式。以打开一个word文档a.doc文件为例,Java中可以有以下两种写法:

exec("start a.doc");

exec(" c:\\Program Files\\MicrosoftOffice\\office winword.exe a.doc");

显然,前一种方法更为简捷方便。

   3、执行一个有标准输出的DOS可执行程序在windows 平台上,运行被调用程序的DOS窗口在程序执行完毕后往往并不会自动关闭,从而导致Java应用程序阻塞在waitfor( )。导致该现象的一个可能的原因是,该可执行程序的标准输出比较多,而运行窗口的标准输出缓冲区不够大。解决的办法是,利用Java提供的Process 类提供的方法让Java虚拟机截获被调用程序的DOS运行窗口的标准输出,在waitfor()命令之前读出窗口的标准输出缓冲区中的内容。一段典型的程序如下:

String str;

Process process =Runtime.getRuntime().exec("cmd /c dir windows");

BufferedReader bufferedReader = newBufferedReader( new InputStreamReader(process.getInputStream()));

while ( (str=bufferedReader.readLine()) !=null) System.out.println(str);  

process.waitfor();

示例

public static boolean resize(String pic,String picTo,int width,int height) {

boolean result = true;

String cmd = "cmd /c convert -sample " + width + "x" + height + " "" + pic + """ +" "" + picTo+""";

log.debug(cmd);

try {

Process process = Runtime.getRuntime().exec(cmd);

if (process.getErrorStream().read() != -1) {

result = false;

process.destroy();

}

} catch (IOException e) {

log.debug("creat icon pic fail!" + e);

result = false;

}

/*BufferedReader bufferedReader = new BufferedReader( newInputStreamReader(process.getInputStream());

while ( (str=bufferedReader.readLine()) != null)System.out.println(str);   */

return result;

}

####################################################################################

我使用上面的程序处理不好使。然后查到下面的blog看到了如下内容。问题被解决。^-^

####################################################################################

Process process = Runtime.getRuntime.exec(cmd); // 执行调用perl命令

InputStream is = process.getInputStream(); // 获取perl进程的输出流

BufferedReader br = new Buffered(new InputStreamReader(is)); // 缓冲读入

StringBuilder buf = new StringBuilder(); // 保存perl的输出结果流

String line = null;

while((line = br.readLine()) != null) buf.append(line); // 循环等待进程结束

System.out.println("ffmpeg输出内容为:" + buf);

……

本来一般都是这样来调用程序并获取进程的输出流的,但是我在windows上执行这样的调用的时候却总是在while那里被堵塞了,结果造成ffmpeg程序在执行了一会后不再执行,这里从官方的参考文档中我们可以看到这是由于缓冲区的问题,由于java进程没有清空ffmpeg程序写到缓冲区的内容,结果导致ffmpeg程序一直在等待。在网上也查找了很多这样的问题,不过说的都是使用单独的线程来进行控制,我也尝试过很多网是所说的方法,可一直没起什么作用。下面就是我的解决方法了,注意到上述代码中的红色部分了么?这里就是关键,我把它改成如下结果就可以正常运行了。

InputStream is = process.getErrorStream(); // 获取ffmpeg进程的输出流

注意到没?我把它改成获取错误流这样进程就不会被堵塞了,而我之前一直想的是同样的命令我手动调用的时候可以完成,而java调用却总是完成不了,一直认为是getInputStream的缓冲区没有被清空,不过问题确实是缓冲区的内容没有被清空,但不是getInputStream的,而是getErrorStream的缓冲区,这样问题就得到解决了。所以我们在遇到java调用外部程序而导致线程阻塞的时候,可以考虑使用两个线程来同时清空process获取的两个输入流,如下这段程序:

……

Process p = Runtime.getRuntime().exec("perl class.pl");

//Process p = Runtime.getRuntime().exec("cmd.exe /c dir");

final InputStream is1 = p.getInputStream();

new Thread(new Runnable() {

public void run() {

BufferedReader br = new BufferedReader(new InputStreamReader(is1));

try{

while(br.readLine() != null) ;

}

catch(Exception e) {

e.printStackTrace();

}

}

}).start(); // 启动单独的线程来清空p.getInputStream()的缓冲区

InputStream is2 = p.getErrorStream();

BufferedReader br2 = new BufferedReader(new InputStreamReader(is2));

StringBuilder buf = new StringBuilder(); // 保存输出结果流

String line = null;

while((line = br2.readLine()) != null) buf.append(line); //

System.out.println("输出结果为:" + buf);

……

通过这样我们使用一个线程来读取process.getInputStream()的输出流,使用另外一个线程来获取process.getErrorStream()的输出流,这样我们就可以保证缓冲区得到及时的清空而不担心线程被阻塞了。当然根据需要你也可以保留process.getInputStream()流中的内容,这个就看调用的程序的处理了。

今天看了斯坦福分词器的源码,发现用了大量System.err.print,怪不得需要使用getErrorStream()捕捉!关于System.err和System.out的区别,可以参考别的日志。这两个流走的是不同的管道。所以需要分别捕捉。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: