調用java.lang.Runtime.exec的正確姿勢分享
調用java.lang.Runtime.exec的正確姿勢
今天寫一個用到編譯的程序,遇到瞭問題。
在調用
runtime.exec("javac HelloWorld.java");
運行完美,也就是有生成.class。
而到瞭
runtime.exec("java HelloWorld >> output.txt");
卻怎麼也無法重定向輸出,連output.txt文件也生成不瞭。
測試”echo hello >> 1.txt” 也是不可以,甚是頭疼,於是乎翻閱資料,這才發現瞭
一個認識上的誤區,就是exec(str)中 不能把str完全看作命令行執行的command。尤其是str中不可包含重定向 ‘ < ‘ ‘ > ‘ 和管道符’ | ‘ 。
那麼,遇到這樣的指令怎麼辦呢?我們接著往下看:
兩種方法
一種是將指令寫到腳本中,在runtime.exec()中調用腳本。這種方法避過瞭使用exec(),也是一種思路。
還有一種方法,就是調用exec()的重載方法:我們來重點看這種方法:
我們先看一下官方doc[>link<]給我們提供的重載方法:
public Process exec(String command) throws IOExecption public Process exec(String command,String [] envp) throws IOExecption public Process exec(String command,String [] envp,File dir) throws IOExecption public Process exec(String[] cmdarray) throws IOExecption public Process exec(String[] cmdarray,String [] envp) throws IOExecption public Process exec(String[] cmdarray,String [] envp,File dir) throws IOExecption
翻閱其文檔,發現其重載方法4.exec(String []cmdarray) 最簡便適合我們,官方說4.exec() 與執行6.exec(cmdarray,null,null) 效果是一樣的。那麼5.exec.(cmdarray,null)也是一樣的咯?
於是乎,我們可以這樣寫:
runtime.exec( new String[]{"/bin/bash", "-c", "java HelloWorld >> output.txt"} ); runtime.exec( new String[]{"/bin/bash", "-c", "java HelloWorld >> output.txt"} ,null ); runtime.exec( new String[]{"/bin/bash", "-c", "java HelloWorld >> output.txt"} ,null,null );
不過要註意,如果使用java /home/path/HelloWorld 時,’ / ‘會被解析成 ” . “,從而報出 “錯誤: 找不到或無法加載主類 .home.path.HelloWorld ”.
所以,無法使用全路徑的時候,我們需要更改一下策略,把 路徑 改到工作目錄dir 中去,比如:
File dir = new File("/home/path/");
然後用其第6種重載方法,把dir作為第三個參數傳入即可:
String []cmdarry ={"/bin/bash", "-c", "java HelloWorld >> output.txt"} runtime.exec(cmdarry,null.dir);
當然echo , ls 等命令便不受’ / ‘限制瞭。
*BTW,exec()取得返回值的標準用法詳見:runtime.exec()的左膀右臂
小結一下
當命令中包含重定向 ‘ < ‘ ‘ > ‘ 和管道符’ | ‘ 時,exec(String command)方法便不適用瞭,需要使用exec(String [] cmdArray) 或者exec(String []cmdarray,String []envp,File dir)來執行。
例如:
exec("echo hello >> ouput.txt"); exec("history | grep -i mvn");
應改為:
exec( new String[]{"/bin/sh","-c","echo hello >> ouput.txt"}); exec( new String[]{"/bin/bash","-c","history | grep -i mvn"},null);
Java Runtime.exec()註意事項
Runtime.exec()
用來執行外部程序或命令
1.Runtime.exec() 有四種調用方法
* public Process exec(String command); * public Process exec(String [] cmdArray); * public Process exec(String command, String [] envp); * public Process exec(String [] cmdArray, String [] envp);
2.得到程序執行返回值, 0為success
需要用waitFor()函數,比如
Process p = Runtime.getRuntime().exec("javac"); (處理.....) int exitVal = p.waitFor();
3.得到程序執行的結果或錯誤信息
需要用BufferedInputStream 和 BufferReader來得到,否則程序會hang
比如得到錯誤信息用p.getErrorStream(),然後輸出即可:
BufferedInputStream in = new BufferedInputStream(p.getErrorStream()); BufferedReader br = new BufferedReader(new InputStreamReader(in));
4.Runtime.exec()
不等同於直接執行command line命令
啊,我算是在這裡吃瞭苦頭瞭。Runtime.exec()很有局限性,對有些命令不能直接把command line裡的內容當作String參數傳給exec().
比如重定向等命令。舉個例子:
javap -l xxx > output.txt
這時要用到exec的第二種重載,即input 參數為String[]:
Process p = Runtime.getRuntime().exec(new String[]{"/bin/sh","-c","javap -l xxx > output.txt"});
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。
推薦閱讀:
- Java調用Shell命令和腳本的實現
- Java魔法堂之調用外部程序的方法
- java.lang.Runtime.exec的左膀右臂:流輸入和流讀取詳解
- 使用Runtime 調用Process.waitfor導致的阻塞問題
- Java Runtime的使用詳解