使用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