java的外部引用_Java 调用外部程序
在Java中可以調用外部程序,這需要通過Process等類來實現。
創建進程
先來介紹一下Process的創建,我們需要使用ProcessBuilder類。如果需要命令行參數的話,則傳入多個參數。比方說下面我就創建了一個查看Java版本號的ProcessBuilder。
ProcessBuilder pb = new ProcessBuilder("java","-version");
ProcessBuilder還有一些成員方法,可以重定向輸入輸出流到文件、設置命令行參數等等。如果需要詳細的使用方法可以參考官方文檔。
有了ProcessBuilder僅僅是第一步,我們還沒有實際執行程序。為了執行程序,我們需要調用它的start()方法,這會啟動進程并返回一個Process對象。如果需要Process的詳細信息,請參考Java官方文檔。
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);
}
總結
以上是生活随笔為你收集整理的java的外部引用_Java 调用外部程序的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: java语言的数组描述_下列关于Java
- 下一篇: java读取frp_【原创】FRP初探(