public String execFlow() {
Process shellProcess = null;
BufferedReader shellErrorResultReader = null;
BufferedReader shellInfoResultReader = null;
int exitCode;
try {
StringBuilder builder = new StringBuilder();
shellProcess = Runtime.getRuntime().exec(new String[]{"/bin/sh", "-c", "需要执行的命令"});
shellErrorResultReader = new BufferedReader(new InputStreamReader(shellProcess.getErrorStream()));
shellInfoResultReader = new BufferedReader(new InputStreamReader(shellProcess.getInputStream()));
String infoLine;
while ((infoLine = shellInfoResultReader.readLine()) != null) {
logger.info("脚本文件执行成功信息:{}", infoLine);
builder.append(infoLine);
}
String errorLine;
while ((errorLine = shellErrorResultReader.readLine()) != null) {
logger.warn("脚本文件执行错误信息:{}", errorLine);
builder.append(errorLine);
}
// 等待程序执行结束并输出状态
exitCode = shellProcess.waitFor();
if (0 == exitCode) {
logger.info("脚本文件执行成功状态:" + exitCode);
} else {
logger.error("脚本文件执行失败状态:" + exitCode);
}
return builder.toString();
} catch (Exception e) {
logger.error("shell脚本执行错误", e);
}finally {
try {
if (shellErrorResultReader != null) {
shellErrorResultReader.close();
}
if (shellInfoResultReader != null) {
shellInfoResultReader.close();
}
if (shellProcess != null) {
shellProcess.destroy();
}
} catch (IOException e) {
e.printStackTrace();
logger.error("关闭Runtime.getRuntime()线程或流失败!");
}
}
return null;
}
注:无法执行切换用户,等需要输入的操作。
如有大佬知晓,麻烦大佬指教
下面的一些解释来源:https://www.cnblogs.com/tohxyblog/p/6501396.html,由于他的排版不好看,我自己重新排一下
java在企业级项目开发中,无论是强制性的功能需要,还是为了简便java的实现,需要调用服务器命令脚本来执行。在java中,RunTime.getRuntime().exec()就实现了这个功能。
举例:
1. RunTime.getRuntime().exec(String command);
在windows下相当于直接调用 /开始/搜索程序和文件 的指令,比如
Runtime.getRuntime().exec("notepad.exe"); -------打开windows下记事本。
Linux下:必须加上/bin/sh,“-c”
Runtime.getRuntime().exec(new String[]{"/bin/sh","-c", ";
Windows下:
Runtime.getRuntime().exec(new String[]{ "cmd", "/c", cmds});
深入:
Process的几种方法:
1.destroy():杀掉子进程
2.exitValue():返回子进程的出口值,值 0 表示正常终止
3.getErrorStream():获取子进程的错误流
4.getInputStream():获取子进程的输入流
5.getOutputStream():获取子进程的输出流
6.waitFor():导致当前线程等待,如有必要,一直要等到由该 Process 对象表示的进程已经终止。如果已终止该子进程,此方法立即返回。如果没有终止该子进程,调用的线程将被阻塞,直到退出子进程,根据惯例,0 表示正常终止
注意:在java中,调用runtime线程执行脚本是非常消耗资源的,所以切忌不要频繁使用!
在调用runtime去执行脚本的时候,其实就是JVM开了一个子线程去调用JVM所在系统的命令,其中开了三个通道: 输入流、输出流、错误流,其中输出流就是子线程走调用的通道。
大家都知道,waitFor是等待子线程执行命令结束后才执行, 但是在runtime中,打开程序的命令如果不关闭,就不算子线程结束。比如以下代码。
代码:
private static Process p = null;
p = Runtime.getRuntime().exec("notepad.exe");
p.waitFor();
System.out.println("-----------------------------我被执行了");
以上代码中,打开windows中记事本。如果我们不手动关闭记事本,那么输出语句就不会被执行,这点是需要理解的。
process的阻塞:
在runtime执行大点的命令中,输入流和错误流会不断有流进入存储在JVM的缓冲区中,如果缓冲区的流不被读取被填满时,就会造成runtime的阻塞。所以在进行比如:大文件复制等的操作时,我们还需要不断的去读取JVM中的缓冲区的流,来防止Runtime的死锁阻塞。