在模擬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(); } } }