Java如何執行操作系統的CMD命令行
阿新 • • 發佈:2019-05-13
終端 erro 方法 數據 lin 調用 encoding reat tar
在模擬cmd調用Python時遇到一些情況,這類問題可以歸類為“超時,阻塞”等,問題原因:
Process 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)。
完整代碼:
import java.io.BufferedInputStream; import java.io.BufferedReader; import java.io.InputStreamReader; /** * Create by [email protected] 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();
}
}
}
Java如何執行操作系統的CMD命令行