在模拟cmd调用Python时遇到一些状况,这类问题能够归类为“超时,阻塞”等,问题缘由:java
Process p=Runtime.getRuntime().exec(String[] cmd);
Runtime.exec方法将产生一个本地的进程,并返回一个Process子类的实例,该实例可用于控制进程或取得进程的相关信息。 因为调用Runtime.exec方法所建立的子进程没有本身的终端或控制台,所以该子进程的标准IO(如stdin,stdou,stderr)都经过p.getOutputStream(),p.getInputStream(),p.getErrorStream() 方法重定向给它的父进程了.用户须要用这些stream来向子进程输入数据或获取子进程的输出。app
例如:Runtime.getRuntime().exec("ls") 另外须要关心的是Runtime.getRuntime().exec()中产生停滞(阻塞,blocking)的问题? 这个是由于Runtime.getRuntime().exec()要本身去处理stdout和stderr的输出, 就是说,执行的结果不知道是现有错误输出(stderr),仍是现有标准输出(stdout)。你没法判断到底那个先输出,因此可能没法读取输出,而一直阻塞。 例如:你先处理标准输出(stdout),可是处理的结果是先有错误输出(stderr), 一直在等错误输出(stderr)被取走了,才到标准输出(stdout),这样就产生了阻塞。ui
解决办法:编码
用两个线程将标准输出(stdout)和错误输出(stderr)。线程
完整代码:code
import java.io.BufferedInputStream; import java.io.BufferedReader; import java.io.InputStreamReader; /** * Create by yster@foxmail.com 2018/11/9 0009 22:28 */ public class ExecuteCmd { /** * 执行外部程序,并获取标准输出 */ public static String execute(String[] cmd,String... encoding) { BufferedReader bufferedReader; InputStreamReader inputStreamReader; try { Process p = Runtime.getRuntime().exec(cmd); /* 为"错误输出流"单独开一个线程读取之,不然会形成标准输出流的阻塞 */ Thread t = new Thread(new InputStreamRunnable(p.getErrorStream(), "ErrorStream")); t.start(); /* "标准输出流"就在当前方法中读取 */ BufferedInputStream bis = new BufferedInputStream(p.getInputStream()); if (encoding != null && encoding.length != 0) { inputStreamReader = new InputStreamReader(bis, encoding[0]);// 设置编码方式 } else { inputStreamReader = new InputStreamReader(bis, "utf-8"); } bufferedReader = new BufferedReader(inputStreamReader); StringBuilder sb = new StringBuilder(); String line; while ((line = bufferedReader.readLine()) != null) { sb.append(line); sb.append("\n"); } bufferedReader.close(); p.destroy(); return sb.toString(); } catch (Exception e) { e.printStackTrace(); return null; } } }
import java.io.BufferedInputStream; import java.io.BufferedReader; import java.io.InputStream; import java.io.InputStreamReader; class InputStreamRunnable implements Runnable { private BufferedReader bReader = null; InputStreamRunnable(InputStream is, String type) { try { bReader = new BufferedReader(new InputStreamReader(new BufferedInputStream(is), "UTF-8")); } catch (Exception ex) { ex.printStackTrace(); } } public void run() { String line; int num = 1; try { while ((line = bReader.readLine()) != null) { //System.out.println("---->"+String.format("%02d",num++)+" "+line); } bReader.close(); } catch (Exception ex) { ex.printStackTrace(); } } }