Java如何执行操作系统的CMD命令行

1,291 阅读2分钟

  在模拟cmd调用Python时遇到一些情况,这类问题可以归类为“超时,阻塞”等,问题原因:

  CopyProcess p=Runtime.getRuntime().exec(String[] cmd);

  Runtime.exec方法将产生一个本地的进程,并返回一个Process子类的实例,该实例可用于控制进程或取得进程的相关信息。 由于调用Runtime.exec方法所创建的子进程没有自己的终端或控制台,因此该子进程的标准IO(如stdin,stdou,stderr)都通过p.getOutputStream(),p.getInputStream(),p.getErrorStream() 方法重定向给它的父进程了.用户需要用这些stream来向子进程输入数据或获取子进程的输出。

  例如:Runtime.getRuntime().exec(ls) 另外需要关心的是Runtime.getRuntime().exec()中产生停滞(阻塞,blocking)的问题? 这个是因为Runtime.getRuntime().exec()要自己去处理stdout和stderr的输出, 就是说,执行的结果不知道是现有错误输出(stderr),还是现有标准输出(stdout)。你无法判断到底那个先输出,所以可能无法读取输出,而一直阻塞。 例如:你先处理标准输出(stdout),但是处理的结果是先有错误输出(stderr), 一直在等错误输出(stderr)被取走了,才到标准输出(stdout),这样就产生了阻塞。

  解决办法:

  用两个线程将标准输出(stdout)和错误输出(stderr)。

  完整代码:

  Copyimport 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;

  }

  }

  }

  Copyimport 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();

  }

  }

  }