調用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。

推薦閱讀: