在Java中可以調(diào)用外部程序,這需要通過Process等類來實現(xiàn)。
創(chuàng)建進(jìn)程
先來介紹一下Process的創(chuàng)建,我們需要使用ProcessBuilder類。如果需要命令行參數(shù)的話,則傳入多個參數(shù)。比方說下面我就創(chuàng)建了一個查看Java版本號的ProcessBuilder。
ProcessBuilder pb = new ProcessBuilder("java","-version");
ProcessBuilder還有一些成員方法,可以重定向輸入輸出流到文件、設(shè)置命令行參數(shù)等等。如果需要詳細(xì)的使用方法可以參考官方文檔。
有了ProcessBuilder僅僅是第一步,我們還沒有實際執(zhí)行程序。為了執(zhí)行程序,我們需要調(diào)用它的start()方法,這會啟動進(jìn)程并返回一個Process對象。如果需要Process的詳細(xì)信息,請參考Java官方文檔。
Process process = pb.start();
這樣的話,命令行對應(yīng)的進(jìn)程就會開始執(zhí)行。我們可以調(diào)用Process的exitValue()方法獲取進(jìn)程是否成功返回(一般返回0為正常退出,記得C語言最后的return 0嗎)。如果需要獲取進(jìn)程的輸出,可以調(diào)用getInputStream()獲取程序的輸入流。需要注意進(jìn)程的輸入輸出和我們Java程序的輸入輸出方向正好是相反的,所以如果我們想要向進(jìn)程中傳遞參數(shù),就需要調(diào)用它的getOutputStream獲取輸出流。
byte[] bytes = new byte[process.getInputStream().available()];
process.getInputStream().read(bytes);
System.out.println(new String(bytes));
進(jìn)程的阻塞
如果你實際執(zhí)行上面的代碼的話,很可能拋出IllegalThreadStateException。因為在我們獲取程序輸出的時候,很有可能當(dāng)前進(jìn)程并沒有結(jié)束。那么獲取結(jié)果就是不合法的操作。因此,為了安全的等待進(jìn)程結(jié)束,我們需要調(diào)用waitFor()方法,阻塞當(dāng)前線程,直到進(jìn)程退出為止。
所以最后的代碼類似這樣。在進(jìn)程啟動之后,我們需要阻塞,直到它結(jié)束。然后獲取返回值和輸出結(jié)果。
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));
輸出流的處理
上面的代碼應(yīng)該沒有問題,而且實際執(zhí)行的時候,返回值為0,。這說明我們確實成功地執(zhí)行了java -version命令。但是,如果你實際執(zhí)行的話,會發(fā)現(xiàn)程序也僅僅輸出了返回值。那么我們期望的實際輸出去哪兒了?
如果研究一下ProcessBuilder的文檔的話,會發(fā)現(xiàn)有這么一個方法redirectErrorStream(boolean),該方法的作用是將子進(jìn)程的錯誤流重定向到標(biāo)準(zhǔn)輸出流上。這樣我們使用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));
結(jié)果會顯示當(dāng)前安裝的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調(diào)用外部進(jìn)程的模板例子。只需要簡單替換一下命令行參數(shù)即可啟動不同的程序。
記事本
調(diào)用notepad就可以啟動記事本了。由于我們調(diào)用了process.waitFor(),所以當(dāng)記事本窗口關(guān)閉前,Java程序也不會關(guān)閉。同理,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));
查看當(dāng)前Windows版本
這個也很簡單,只需要在命令提示符窗口中輸入ver即可。但是我們不能直接將進(jìn)程名寫為ver。因為實際上沒有這個程序,這只是命令提示符的功能而已。所以代碼要修改一下,我們調(diào)用的進(jìn)程實際上是cmd,參數(shù)是ver。
另外默認(rèn)編碼是UTF-8,而在中文操作系統(tǒng)下編碼是GBK。所以會出現(xiàn)亂碼。所以輸出流的代碼也需要修改,我們將它包裝到BufferedReader中,BufferedReader有一個接受字符集參數(shù)的構(gòu)造方法。而且BufferedReader在Java 8中還新增了一個lines()方法,返回所有輸入行的stream,我們可以利用Java 8的流類庫和lambda表達(dá)式方便的處理。
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);
}