java使用Runtime执行系统命令遇到的问题

使用Runtime执行系统命令遇到的问题

java执行系统命令的工具,Runtime.getRuntime().exec(),这个工具里面的坑还是不少的,大部分网上已经分析了,我遇到一个没有在网上找到的。

首先说一下这个坑,在command中如果参数的内容中有空格、tab、换行符、回车符,就会执行失败,例如:grep "abc def" -r -n ./这是因为exec方法使用的是StringTokenizer对command分割的。

参数含义

  • command:要执行的命令。
  • envp:设置环境变量,值是:环境变量名=环境变量值,例如 JAVA_HOEM=/usr/lib/java1.8
  • dir:设置命令执行的目录
    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*"});

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

你可能感兴趣的