Java 調用外部程序

MonikaBarba 7年前發布 | 16K 次閱讀 Java Java開發

在Java中可以調用外部程序,這需要通過Process等類來實現。

創建進程

先來介紹一下Process的創建,我們需要使用 ProcessBuilder 類。如果需要命令行參數的話,則傳入多個參數。比方說下面我就創建了一個查看Java版本號的 ProcessBuilder 。

ProcessBuilder pb = new ProcessBuilder("java","-version");

ProcessBuilder 還有一些成員方法,可以重定向輸入輸出流到文件、設置命令行參數等等。

有了 ProcessBuilder 僅僅是第一步,我們還沒有實際執行程序。為了執行程序,我們需要調用它的 start() 方法,這會啟動進程并返回一個 Process 對象。

Process process = pb.start();

這樣的話,命令行對應的進程就會開始執行。我們可以調用 Process 的 exitValue() 方法獲取進程是否成功返回(一般返回0為正常退出,記得C語言最后的 return 0 嗎)。如果需要獲取進程的輸出,可以調用 getInputStream() 獲取程序的輸入流。需要注意進程的輸入輸出和我們Java程序的輸入輸出方向正好是相反的,所以如果我們想要向進程中傳遞參數,就需要調用它的 getOutputStream 獲取輸出流。

byte[] bytes = new byte[process.getInputStream().available()];
process.getInputStream().read(bytes);
System.out.println(new String(bytes));

進程的阻塞

如果你實際執行上面的代碼的話,很可能拋出 IllegalThreadStateException 。因為在我們獲取程序輸出的時候,很有可能當前進程并沒有結束。那么獲取結果就是不合法的操作。因此,為了安全的等待進程結束,我們需要調用 waitFor() 方法,阻塞當前線程,直到進程退出為止。

所以最后的代碼類似這樣。在進程啟動之后,我們需要阻塞,直到它結束。然后獲取返回值和輸出結果。

ProcessBuilder pb = new ProcessBuilder("java","-version");
Process process = pb.start();
process.waitFor();
System.out.println(process.exitValue());
byte[] bytes = new byte[process.getInputStream().available()];
process.getInputStream().read(bytes);
System.out.println(new String(bytes));

輸出流的處理

上面的代碼應該沒有問題,而且實際執行的時候,返回值為0,。這說明我們確實成功地執行了 java -version 命令。但是,如果你實際執行的話,會發現程序也僅僅輸出了返回值。那么我們期望的實際輸出去哪兒了?

如果研究一下 ProcessBuilder 的文檔的話,會發現有這么一個方法 redirectErrorStream(boolean) ,該方法的作用是將子進程的錯誤流重定向到標準輸出流上。這樣我們使用 Process.getInputStream() 就可以獲取到所有輸出了。

所以最后的代碼如下。

ProcessBuilder pb = new ProcessBuilder("java","-version");
pb.redirectErrorStream(true);
Process process = pb.start();
process.waitFor();
System.out.println(process.exitValue());
byte[] bytes = new byte[process.getInputStream().available()];
process.getInputStream().read(bytes);
System.out.println(new String(bytes));

結果會顯示當前安裝的Java版本號信息。

0
java version "1.8.0_121"
Java(TM) SE Runtime Environment (build 1.8.0_121-b13)
Java HotSpot(TM) 64-Bit Server VM (build 25.121-b13, mixed mode)

其他例子

通過一番研究,我們得到了Java調用外部進程的模板例子。只需要簡單替換一下命令行參數即可啟動不同的程序。

記事本

調用 notepad 就可以啟動記事本了。由于我們調用了 process.waitFor() ,所以當記事本窗口關閉前,Java程序也不會關閉。同理, calc 可以啟動計算器, explorer 可以啟動資源管理器。

ProcessBuilder pb = new ProcessBuilder("notepad");
pb.redirectErrorStream(true);
Process process = pb.start();
process.waitFor();
System.out.println(process.exitValue());
byte[] bytes = new byte[process.getInputStream().available()];
process.getInputStream().read(bytes);
System.out.println(new String(bytes));

查看當前Windows版本

這個也很簡單,只需要在命令提示符窗口中輸入 ver 即可。但是我們不能直接將進程名寫為 ver 。因為實際上沒有這個程序,這只是命令提示符的功能而已。所以代碼要修改一下,我們調用的進程實際上是 cmd ,參數是 ver 。

另外默認編碼是UTF-8,而在中文操作系統下編碼是GBK。所以會出現亂碼。所以輸出流的代碼也需要修改,我們將它包裝到 BufferedReader 中, BufferedReader 有一個接受字符集參數的構造方法。而且 BufferedReader 在Java 8中還新增了一個 lines() 方法,返回所有輸入行的 stream ,我們可以利用Java 8的流類庫和lambda表達式方便的處理。

ProcessBuilder pb = new ProcessBuilder("cmd", "/c", "ver");
pb.redirectErrorStream(true);
Process process = pb.start();
process.waitFor();
System.out.println(process.exitValue());
try (BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream(), "GBK"))) {
    String result = reader.lines()
            .collect(Collectors.joining("\n"));
    System.out.println(result);
}

 

來自:http://www.jianshu.com/p/428bbc3852a7

 

 本文由用戶 MonikaBarba 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!