1. 程式人生 > >Java如何執行操作系統的CMD命令行

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

終端 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命令行