在java中,RunTime.getRuntime().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)----在指定环境的独立进程中执行指定的命令和变量

public Process exec(String command,String[] envp,File dir)----在有指定环境和工作目录的独立进程中执行指定的字符串命令

public Process exec(String[] cmdarray,String[] envp,File dir)----在指定环境和工作目录的独立进程中执行指定的命令和变量

 

举例:

1.  RunTime.getRuntime().exec(String  command);

            在windows下相当于直接调用   /开始/搜索程序和文件  的指令,比如

            Runtime.getRuntime().exec("notepad.exe");  -------打开windows下记事本。

2.  public Process exec(String [] cmdArray);

             Linux下:

             Runtime.getRuntime().exec(new String[]{"/bin/sh","-c", ";

             Windows下:

             Runtime.getRuntime().exec(new String[]{ "cmd", "/c", cmds});

 

实例:

String command = "find " + source.getRoute() + " -name '" +source.getName();   

Process process = Runtime.getRuntime().exec(new String[] {"/bin/sh","-c",command});

 

补充:#!/bin/bash和#!/bin/sh的区别

#! 是个指示路径的表示符,/bin/bash和/bin/sh指定了脚本解析器的程序路径

bash是sh的完整版,bash完全兼容sh命令,反之不行

OPTIONS:

     -c string      该选项表明string中包含了一条命令.如 bash -c ls ~

  -i       使Bash以交互式方式运行

  -r       使Bash以受限方式运行

  --login       使Bash以登录Shell方式运行

  --posix      使Bash遵循POSIX标准

  --verbose    使Bash显示所有其读入的输入行

  --help     打印Bash的使用信息

  --version     打印版本信息

以上参考链接:https://www.jianshu.com/p/f3b8b4ad5bf8
以下参考链接:https://www.jianshu.com/p/43ce90abb5cd

三、经验总结

Runtime run = Runtime.getRuntime();
Process p = run.exec("ping 127.0.0.1");
这是基本用法,通过p可以获得命令执行过程中的输入流和错误流(也就是打印信息)
InputStream ins= p.getInputStream();
InputStream ers= p.getErrorStream();
这两个流要用不同的线程去处理,不然容易引发阻塞
当默认的路径(java代码执行默认为class路径)执行bat命令时,如果没有跨盘符,通过bat文件的绝对路径或者相对路径可以执行到对应的bat命令,如果跨盘符,就必须设置执行文件路径了,即需要用到如下命令 public Process exec(String command,String[] envp,File dir),例如:Process p = run.exec("ceshi.bat",null,new File("d:\ceshi\"));或者Process p = run.exec(new String[]{"cmd","/c","start","ceshi.bat"},null,new File("d:\ceshi\"));

如果命令执行时报“java.io.IOException: Cannot run program "****": CreateProcess error=2, ϵͳÕҲ»µ½ָ¶”这样的错误,表明命令语法错误,即命令不能被正确识别,这个需要仔细检查命令语法,尤其是要注意有空格的长命令,列如run.exec("%comspec% /c \ceshi\ceshi.bat");此命令需要run.exec(new String[]{"cmd","/c","%comspec% /c \ceshi\ceshi.bat"},null,null);

如果执行时报“java.lang.IllegalArgumentException: Executable name has embedded quote,split the arguments”的异常,很可能是命令中有空格识别异常,可以试着把空格转成带引号的空格,列如run.exec("c:\ceshi file\ceshi.bat".replace(" ","" ""))

如果发现一个bat命令在CMD窗口中可以正常执行,但通过java的Runtime执行就会阻塞,八成可能是命令执行中开了新窗口,但老窗口没关导致的,列如run.exec(new String[]{"cmd","/k","ceshi.bat"},null,new File("d:\ceshi\"));这条语句在执行的过程中因为存在“/k”的存在,导致命令执行完dir命令后不关闭命令窗口,在获取流的时候,流一直不关闭,就会形成阻塞,只要把“/k”换成“/c”就行了

如果遇到需要执行两个bat文件命令bat1、bat2,其中bat2依赖与bat1的环境,即在CMD窗口中,先执行bat1后再在同一个窗口中执行bat2,就很正常,但如果在两个不同的CMD窗口执行这两个命令,就只有bat1执行成功,这个解决办法很简单,只要新建一个bat文件,然后通过“call”命令来调用这两个bat文件,就不会出错了(这有个知识点:start调用一个bat命令会新启一个CMD窗口,call不会新开窗口,而是在当前窗口执行调用的bat文件命令,执行完后,继续执行下面的命令,所以start雷士java中的多线程启动,call类似java中简单的函数调用)

四、示例代码:

 

public class RuntimeTest {
    public static void main(String[] args) {
        RuntimeTest s = new RuntimeTest();
        s.test();
    }
    public void test(){
        Runtime run =Runtime.getRuntime();
        try {
            Process p = run.exec("ping 127.0.0.1");
            InputStream ins= p.getInputStream();
            InputStream ers= p.getErrorStream();
            new Thread(new inputStreamThread(ins)).start();
            p.waitFor();
        } catch (IOException e) {
            e.printStackTrace();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
    class inputStreamThread implements Runnable{
        private InputStream ins = null;
        private BufferedReader bfr = null;
        public inputStreamThread(InputStream ins){
            this.ins = ins;
            this.bfr = new BufferedReader(new InputStreamReader(ins));
        }
        @Override
        public void run() {
            String line = null;
            byte[] b = new byte[100];
            int num = 0;
            try {
                while((num=ins.read(b))!=-1){
                    System.out.println(new String(b,"gb2312"));
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}



 

Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐