使用Runtime執行系統命令遇到的問題
java執行系統命令的工具,Runtime.getRuntime().exec(),這個工具里面的坑還是不少的,大部分網上已經分析了,我遇到一個沒有在網上找到的。
首先說一下這個坑,在command中如果參數的內容中有空格、tab、換行符、回車符,就會執行失敗,例如:grep "abc def" -r -n ./這是因為exec方法使用的是StringTokenizer對command分割的。
參數含義
-
command
:要執行的命令。 -
envp
:設置環境變量,值是:環境變量名=環境變量值,例如 JAVA_HOEM=/usr/lib/java1.8 -
dir
:設置命令執行的目錄
1
2
3
4
5
6
7
8
9
10
11
|
public Process exec(String command, String[] envp, File dir) throws IOException { if (command.length() == 0 ) throw new IllegalArgumentException( "Empty command" ); StringTokenizer st = new StringTokenizer(command); String[] cmdarray = new String[st.countTokens()]; for ( int i = 0 ; st.hasMoreTokens(); i++) cmdarray[i] = st.nextToken(); return exec(cmdarray, envp, dir); } |
為了解決這個問題我自己寫了一個可以自定義分隔符的類。
這個類中的第二個exec可以自定義分割符,但是也有一個需要注意的地方
假如分隔符是“&&” 要執行的命令這樣寫 : grep &&"abc&&def"&&-r&&-n&&./ 執行也會報錯,以為grep后面有個空格,它會把grep加空格當成一個整體作為一個命令執行,就會出現命令找不到,但是報的錯誤日志如果不注意很容易忽略這個空格,就只能一直苦惱。所以正確的方式就是把所有默認分隔符全都刪掉,只用自己的分隔符就可以了。
下面是實現代碼
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
|
import java.io.File; import java.io.IOException; import java.util.StringTokenizer; public class GxExec { private static GxExec gxExec = new GxExec(); public static GxExec getGxExec() { return gxExec; } private GxExec() {} public Process exec(String[] cmdarray, String[] envp, File dir) throws IOException { return new ProcessBuilder(cmdarray) .start(); } public Process exec(String command, String delim) throws IOException { if (command.length() == 0 ) { throw new IllegalArgumentException( "Empty command" ); } StringTokenizer st = new StringTokenizer(command, delim); String[] cmdarray = new String[st.countTokens()]; for ( int i = 0 ; st.hasMoreTokens(); i++) { cmdarray[i] = st.nextToken(); System.out.println(cmdarray[i]); } return exec(cmdarray, null , null ); } } |
java Runtime.exec() 執行
-
Runtime.exec()
不等同于直接執行command line命令! -
Runtime.exec()
很有局限性,對有些命令不能直接把command line里的內容當作String參數傳給exec().
比如重定向等命令。
舉個例子
1
|
javap -l xxx > output.txt |
這時要用到exec的第二種重載,即input 參數為String[]:
1
2
3
|
Process p = Runtime.getRuntime().exec( new String[]{ "/bin/sh" , "-c" , "javap -l xxx > output.txt" }); rm -rf name* Process p = Runtime.getRuntime().exec( new String[]{ "/bin/sh" , "-c" , "rm -rf name*" }); |
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持服務器之家。
原文鏈接:https://blog.csdn.net/zpzkitt/article/details/85679489