Java如何執行操做系統的CMD命令行

在模擬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();
        }
    }
}
相關文章
相關標籤/搜索