在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();
}
}
}
}