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