使用Runtime执行系统命令遇到的问题
java执行系统命令的工具,Runtime.getRuntime().exec(),这个工具里面的坑还是不少的,大部分网上已经分析了,我遇到一个没有在网上找到的。
首先说一下这个坑,在command中如果参数的内容中有空格、tab、换行符、回车符,就会执行失败,例如:grep "abc def" -r -n ./这是因为exec方法使用的是StringTokenizer对command分割的。
参数含义
command
:要执行的命令。envp
:设置环境变量,值是:环境变量名=环境变量值,例如 JAVA_HOEM=/usr/lib/java1.8dir
:设置命令执行的目录
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加空格当成一个整体作为一个命令执行,就会出现命令找不到,但是报的错误日志如果不注意很容易忽略这个空格,就只能一直苦恼。所以正确的方式就是把所有默认分隔符全都删掉,只用自己的分隔符就可以了。
下面是实现代码
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().
比如重定向等命令。
举个例子
javap -l xxx > output.txt
这时要用到exec的第二种重载,即input 参数为String[]:
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*"});
以上为个人经验,希望能给大家一个参考,也希望大家多多支持编程网。