Java 調(diào)用外部程序

在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)用ProcessexitValue()方法獲取進(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);
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,628評論 19 139
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法,類相關(guān)的語法,內(nèi)部類的語法,繼承相關(guān)的語法,異常的語法,線程的語...
    子非魚_t_閱讀 34,740評論 18 399
  • 十多年前,因為L領(lǐng)導(dǎo)把我一個人當(dāng)三個人使喚的原因,所以,我除了負(fù)責(zé)文秘、后勤、行政、工資、考勤的工作以外,我還要協(xié)...
    D018李靜閱讀 296評論 12 15
  • 今天早上和室友們?nèi)チ酥醒氪蠼趾退鞣苼啞?雖然路途波折,大家也沒用埋怨我,很高興 看到中央書店的爺爺們和姐姐們,突然...
    小九亭閱讀 227評論 0 0
  • CHOCK虹閱讀 311評論 3 0

友情鏈接更多精彩內(nèi)容