Java 调用 shell 脚本详解
这一年的项目中,有大量的场景需要Java 进程调用 Linux的bash shell 脚本实现相关功能。
从之前的项目中拷贝的相关模块和网上的例子来看,有个别的“陷阱”造成调用shell 脚本在某些特殊的场景下,有一些奇奇怪怪的bug。
大家且听我一一道来。
先看看网上搜索到的例子:
- package someTest;
- import java.io.BufferedReader;
- import java.io.IOException;
- import java.io.InputStreamReader;
- public class ShellTest {
- public static void main(String[] args) {
- InputStreamReader stdISR = null;
- InputStreamReader errISR = null;
- Process process = null;
- String command = "/home/Lance/workspace/someTest/testbash.sh";
- try {
- process = Runtime.getRuntime().exec(command);
- int exitValue = process.waitFor();
- String line = null;
- stdISR = new InputStreamReader(process.getInputStream());
- BufferedReader stdBR = new BufferedReader(stdISR);
- while ((line = stdBR.readLine()) != null) {
- System.out.println("STD line:" + line);
- }
- errISR = new InputStreamReader(process.getErrorStream());
- BufferedReader errBR = new BufferedReader(errISR);
- while ((line = errBR.readLine()) != null) {
- System.out.println("ERR line:" + line);
- }
- } catch (IOException | InterruptedException e) {
- e.printStackTrace();
- } finally {
- try {
- if (stdISR != null) {
- stdISR.close();
- }
- if (errISR != null) {
- errISR.close();
- }
- if (process != null) {
- process.destroy();
- }
- } catch (IOException e) {
- System.out.println("正式执行命令:" + command + "有IO异常");
- }
- }
- }
- }
testbash.sh
- #!/bin/bash
- echo `pwd`
输出结果为:
- STD line:/home/Lance/workspace/someTest
Java在执行Runtime.getRuntime().exec(command)之后,Linux会创建一个进程,该进程与JVM进程建立三个管道连接,标准输入流、标准输出流、标准错误流。
上述代码,依次读取标准输出流和标准错误流,在shell给出“退出信号”后,做了相应的清理工作。
对于一般场景来说,这段代码可以凑合用了。但是,在实际场景中,会有以下几个“陷阱”。
一. 当标准输出流或标准错误流非常庞大的时候,会出现调用waitFor方法卡死的bug。
真实的环境中,当标准输出在10000行左右的时候,就会出现卡死的情况。
原因分析:假设linux进程不断向标准输出流和标准错误流写数据,而JVM却不读取,数据会暂存在linux缓存区,当缓存区存满之后导致该进程无法继续写数据,会僵死,导致java进程会卡死在waitFor()处,永远无法结束。
解决方式:由于标准输出和错误输出都会向Linux缓存区写数据,而脚本如何输出这两种流是Java端不能确定的。为了不让shell脚本的子进程卡死,这两种输出需要分别读取,而且不能互相影响。所以必须新开两个线程来进行读取。
我开始的实现如下:
- package someTest;
- import java.io.BufferedReader;
- import java.io.IOException;
- import java.io.InputStream;
- import java.io.InputStreamReader;
- import java.util.LinkedList;
- import java.util.List;
- public class CommandStreamGobbler extends Thread {
- private InputStream is;
- private String command;
- private String prefix = "";
- private boolean readFinish = false;
- private boolean ready = false;
- private List<String> infoList = new LinkedList<String>();
- CommandStreamGobbler(InputStream is, String command, String prefix) {
- this.is = is;
- this.command = command;
- this.prefix = prefix;
- }
- public void run() {
- InputStreamReader isr = null;
- try {
- isr = new InputStreamReader(is);
- BufferedReader br = new BufferedReader(isr);
- String line = null;
- ready = true;
- while ((line = br.readLine()) != null) {
- infoList.add(line);
- System.out.println(prefix + " line: " + line);
- }
- } catch (IOException ioe) {
- System.out.println("正式执行命令:" + command + "有IO异常");
- } finally {
- try {
- if (isr != null) {
- isr.close();
- }
- } catch (IOException ioe) {
- System.out.println("正式执行命令:" + command + "有IO异常");
- }
- readFinish = true;
- }
- }
- public InputStream getIs() {
- return is;
- }
- public String getCommand() {
- return command;
- }
- public boolean isReadFinish() {
- return readFinish;
- }
- public boolean isReady() {
- return ready;
- }
- public List<String> getInfoList() {
- return infoList;
- }
- }
- package someTest;
- import java.io.IOException;
- import java.io.InputStreamReader;
- public class ShellTest {
- public static void main(String[] args) {
- InputStreamReader stdISR = null;
- InputStreamReader errISR = null;
- Process process = null;
- String command = "/home/Lance/workspace/someTest/testbash.sh";
- try {
- process = Runtime.getRuntime().exec(command);
- CommandStreamGobbler errorGobbler = new CommandStreamGobbler(process.getErrorStream(), command, "ERR");
- CommandStreamGobbler outputGobbler = new CommandStreamGobbler(process.getInputStream(), command, "STD");
- errorGobbler.start();
- // 必须先等待错误输出ready再建立标准输出
- while (!errorGobbler.isReady()) {
- Thread.sleep(10);
- }
- outputGobbler.start();
- while (!outputGobbler.isReady()) {
- Thread.sleep(10);
- }
- int exitValue = process.waitFor();
- } catch (IOException | InterruptedException e) {
- e.printStackTrace();
- } finally {
- try {
- if (stdISR != null) {
- stdISR.close();
- }
- if (errISR != null) {
- errISR.close();
- }
- if (process != null) {
- process.destroy();
- }
- } catch (IOException e) {
- System.out.println("正式执行命令:" + command + "有IO异常");
- }
- }
- }
- }
到此为止,解决了Java卡死shell脚本的情况。再说说,第二种可能。
二. 由于shell脚本的编写问题,当其自身出现僵死的情况,上述代码出现Java代码被僵死的Shell脚本阻塞住的情况。
原因分析:由于shell脚本也是人写的,难免会出现失误。在Java调用shell脚本时,无论是Debug场景还是生产环境,都发生过shell脚本意外僵死反过来卡死Java相关线程的情况。典型的表现为:shell脚本长时间运行,标准输出和错误输出没有任何输出(包括结束符),操作系统显示shell脚本在正常运行或僵死,没有退出信号。
解决方式:上述代码中,至少有三处会导致线程阻塞,包括标准输出和错误输出这线程的BufferedReader的readline方法,以及Process的waitFor方法。解决这个问题的核心有两个,1.避免任何Java线程被阻塞住,因为一旦被IO阻塞住,线程将处于内核态,主线程没有任何办法强制结束相关子线程。2.添加一个简单的超时机制,超时后回收相应的线程资源,并结束调用过程。
演示代码中,我改写了testshell.sh,写一个没有任何输出的死循环模拟shell卡死的情况。
- #!/bin/bash
- while true;do
- a=1
- sleep 0.1
- done
- package someTest;
- import java.io.BufferedReader;
- import java.io.IOException;
- import java.io.InputStream;
- import java.io.InputStreamReader;
- import java.util.LinkedList;
- import java.util.List;
- public class CommandStreamGobbler extends Thread {
- private InputStream is;
- private String command;
- private String prefix = "";
- private boolean readFinish = false;
- private boolean ready = false;
- // 命令执行结果,0:执行中 1:超时 2:执行完成
- private int commandResult = 0;
- private List<String> infoList = new LinkedList<String>();
- CommandStreamGobbler(InputStream is, String command, String prefix) {
- this.is = is;
- this.command = command;
- this.prefix = prefix;
- }
- public void run() {
- InputStreamReader isr = null;
- BufferedReader br = null;
- try {
- isr = new InputStreamReader(is);
- br = new BufferedReader(isr);
- String line = null;
- ready = true;
- while (commandResult != 1) {
- if (br.ready() || commandResult == 2) {
- if ((line = br.readLine()) != null) {
- infoList.add(line);
- } else {
- break;
- }
- } else {
- Thread.sleep(100);
- }
- }
- } catch (IOException | InterruptedException ioe) {
- System.out.println("正式执行命令:" + command + "有IO异常");
- } finally {
- try {
- if (br != null) {
- br.close();
- }
- if (isr != null) {
- isr.close();
- }
- } catch (IOException ioe) {
- System.out.println("正式执行命令:" + command + "有IO异常");
- }
- readFinish = true;
- }
- }
- public InputStream getIs() {
- return is;
- }
- public String getCommand() {
- return command;
- }
- public boolean isReadFinish() {
- return readFinish;
- }
- public boolean isReady() {
- return ready;
- }
- public List<String> getInfoList() {
- return infoList;
- }
- public void setTimeout(int timeout) {
- this.commandResult = timeout;
- }
- }
- package someTest;
- public class CommandWaitForThread extends Thread {
- private Process process;
- private boolean finish = false;
- private int exitValue = -1;
- public CommandWaitForThread(Process process) {
- this.process = process;
- }
- public void run() {
- try {
- this.exitValue = process.waitFor();
- } catch (InterruptedException e) {
- e.printStackTrace();
- } finally {
- finish = true;
- }
- }
- public boolean isFinish() {
- return finish;
- }
- public void setFinish(boolean finish) {
- this.finish = finish;
- }
- public int getExitValue() {
- return exitValue;
- }
- }
- package someTest;
- import java.io.IOException;
- import java.io.InputStreamReader;
- import java.util.Date;
- public class ShellTest {
- public static void main(String[] args) {
- InputStreamReader stdISR = null;
- InputStreamReader errISR = null;
- Process process = null;
- String command = "/home/Lance/workspace/someTest/testbash.sh";
- long timeout = 10 * 1000;
- try {
- process = Runtime.getRuntime().exec(command);
- CommandStreamGobbler errorGobbler = new CommandStreamGobbler(process.getErrorStream(), command, "ERR");
- CommandStreamGobbler outputGobbler = new CommandStreamGobbler(process.getInputStream(), command, "STD");
- errorGobbler.start();
- // 必须先等待错误输出ready再建立标准输出
- while (!errorGobbler.isReady()) {
- Thread.sleep(10);
- }
- outputGobbler.start();
- while (!outputGobbler.isReady()) {
- Thread.sleep(10);
- }
- CommandWaitForThread commandThread = new CommandWaitForThread(process);
- commandThread.start();
- long commandTime = new Date().getTime();
- long nowTime = new Date().getTime();
- boolean timeoutFlag = false;
- while (!commandIsFinish(commandThread, errorGobbler, outputGobbler)) {
- if (nowTime - commandTime > timeout) {
- timeoutFlag = true;
- break;
- } else {
- Thread.sleep(100);
- nowTime = new Date().getTime();
- }
- }
- if (timeoutFlag) {
- // 命令超时
- errorGobbler.setTimeout(1);
- outputGobbler.setTimeout(1);
- System.out.println("正式执行命令:" + command + "超时");
- }else {
- // 命令执行完成
- errorGobbler.setTimeout(2);
- outputGobbler.setTimeout(2);
- }
- while (true) {
- if (errorGobbler.isReadFinish() && outputGobbler.isReadFinish()) {
- break;
- }
- Thread.sleep(10);
- }
- } catch (IOException | InterruptedException e) {
- e.printStackTrace();
- } finally {
- if (process != null) {
- process.destroy();
- }
- }
- }
- private boolean commandIsFinish(CommandWaitForThread commandThread, CommandStreamGobbler errorGobbler, CommandStreamGobbler outputGobbler) {
- if (commandThread != null) {
- return commandThread.isFinish();
- } else {
- return (errorGobbler.isReadFinish() && outputGobbler.isReadFinish());
- }
- }
- }
在以上的代码中,为了防止线程被阻塞,要点如下:
1. 在CommandStreamGobbler里,bufferedReader在readLine()之前,先用ready()看一下当前缓冲区的情况,请特别注意ready()描述,这个方法是非阻塞的。
- boolean java.io.BufferedReader.ready() throws IOException
- Tells whether this stream is ready to be read. A buffered character stream is ready if the buffer is not empty, or if the underlying character stream is ready.
- Returns:
- True if the next read() is guaranteed not to block for input, false otherwise. Note that returning false does not guarantee that the next read will block.
2.在一个新线程commandThread中,调用process对象的waitFor()从而避免主线程卡死,主线程的最后会执行finally块中的process.destory()保证commandThread正常退出。
以上的两点改进,保证了Java在调用shell脚本过程互不被对方卡死的机制。
三.在执行shell脚本过程中,可能会添加参数,通常在终端中,我们使用“ ”(空格)把参数隔开。
为了区分空格是作为参数分隔符,还是参数的一部分。调用exec方法有特别的注意事项。
- String command = "/home/Lance/workspace/someTest/testbash.sh 'hello world'";
- process = Runtime.getRuntime().exec(command);
等价于
- List<String> commandList = new LinkedList<String>();
- commandList.add("/home/Lance/workspace/someTest/testbash.sh");
- commandList.add("hello world");
- String[] commands = new String[commandList.size()];
- for (int i = 0; i < commandList.size(); i++) {
- commands[i] = commandList.get(i);
- }
- process = Runtime.getRuntime().exec(commands);
好了,今天介绍到这里。
Java 调用 shell 脚本详解的更多相关文章
- 阿里语音识别(语音转文字)java调用全程手把手详解-适合中小学生快速上手
阿里语音识别服务java调用全程手把手详解-适合中小学生快速上手 阿里语音识别与百度语音识别的调用对比: 用例:1分30秒的录音文件 百度用时:3秒 阿里用时:30秒 识别准确率来看 ...
- java调用shell脚本小demo
复制指定文件cpp.sh: [root@localhost soft]# vim cpp.sh#!/bin/bash name="$1"\cp /home/soft/test/${ ...
- [转载]JAVA调用Shell脚本
FROM:http://blog.csdn.net/jj12345jj198999/article/details/11891701 在实际项目中,JAVA有时候需要调用C写出来的东西,除了JNI以外 ...
- java调用shell脚本,并获得结果集的例子
/** * 运行shell脚本 * @param shell 需要运行的shell脚本 */ public static void execShell(String shell){ try { Run ...
- java调用shell脚本
/** * 运行shell脚本 * @param shell 需要运行的shell脚本 */ public static void execShell(String shell){ try { Run ...
- JAVA调用shell脚本利用ansible修改多节点上的redis参数
创建hosts文件 创建ansible-playbook执行时所用到的hosts文件,例如 /etc/redis/hosts 利用shell命令根据传入的host名和地址写入hosts文件: #set ...
- java调用shell脚本执行操作
//定时清空 日志 String shellString = "sh /home/jyapp/delete_log.sh"; Process process = Runtime.g ...
- Java 执行Shell脚本指令
一.介绍 有时候我们在Linux中运行Java程序时,需要调用一些Shell命令和脚本.而Runtime.getRuntime().exec()方法给我们提供了这个功能,而且Runtime.getRu ...
- 用java代码调用shell脚本执行sqoop将hive表中数据导出到mysql
1:创建shell脚本 touch sqoop_options.sh chmod 777 sqoop_options.sh 编辑文件 特地将执行map的个数设置为变量 测试 可以java代码传参数 ...
随机推荐
- SRS流媒体服务器搭建+ffmpeg推流VLC取流观看
一.编译SRS https://github.com/winlinvip/simple-rtmp-server 目前有1.0-release.2.0.3.0等版本 2.0官方文档地址:https:// ...
- JavaScript学习:取数组中最大值和最小值
在实际业务中有的时候要取出数组中的最大值或最小值.但在数组中并没有提供arr.max()和arr.min()这样的方法.那么是不是可以通过别的方式实现类似这样的方法呢?那么今天我们就来整理取出数组中最 ...
- 山寨版 WP8.1 Cortana 启动 PC
8.1 dev preview 发布以来 Cortana 很受关注 前一段看到有视频演示用 Cortana 来启动 PC 看视频也是启动第三方应用实现的,简单来弄其实就是个语音启动应用 + 网络唤醒么 ...
- Vs2017获取Git空仓库后创建解决方案及项目无法推送,推送失败的问题.
与Git无关,因为远程是空文件夹,导致没有远程版本做对应提示更改或怎样,必须在创建人创建仓库的时候上传文件代码. https://developercommunity.visualstudio.c ...
- .NET Core在安装(VS2015)与部署
.NET Core开发环境搭建 使用VS2015开发.NET Core项目,环境的搭建可以参考官网,大致安装步骤如下: 1.首先你得装个vs2015 并且保证已经升级至 update3及以上,下载链接 ...
- C语言关于进制转换,补码, 整数的位操作
菜单导航: 1.二进制.八进制.十进制.十六进制的相互转换 2.原码.反码.补码 3.举例证明整数在计算机内是以补码的形式存在(以负数为例) 4.整数的位操作:按位且&.或|.异或^.取反~ ...
- 【转】ArcGIS10.0完全卸载全攻略
ArcGIS10.0完全卸载详细步骤: 1.开始>控制面板>添加删除程序,卸载所有ArcGIS软件和帮助文档,以及所有ArcGIS补丁.2.从添加删除程序面板中删除所有Python相关的应 ...
- 创建自己的composer包
需求:在项目中输入 p($arr); 将会格式化输出 一.在GitHub上创建仓库 1.1这个仓库必须包含composer.json文件,内容如下. { "name": " ...
- Python线程同步
线程执行 join与setDaemon 子线程在主线程运行结束后,会继续执行完,如果给子线程设置为守护线程(setDaemon=True),主线程运行结束子线程即结束: 如果join()线程,那么主线 ...
- 使用json web token
由来 做了这么长时间的web开发,从JAVA EE中的jsf,spring,hibernate框架,到spring web MVC,到用php框架thinkPHP,到现在的nodejs,我自己的看法是 ...