背景
使用java实现的web端,web端相应用户的界面操作,使用java调用bash实现的shell脚本进行实际的操作,操作完成返回执行结果给web 界面显示。
现象:
Java 进程阻塞。使用ps命令可以看到被调用的shell 的执行进程状态为S
![img](file:///D:\Users\Frank\AppData\Local\Temp\ksohtml\wps52C1.tmp.jpg)
分析
Shell子进程的状态是S 睡眠状态,也就是该进程在等待某个条件满足,方能继续执行。
Java程序在调用Runtime.getRuntime().exec(jyName)之后,linux 会创建一个进程用于执行这个程序,该进程与jvm之间使用三个管道进行链接 标准输入、标准输出、标准出错。
假设这个子进程一直向标准输出或者标准出错中写数据而jvm一致不读取,若相应的标准出错标准输出缓存区被写满,则该子进程会一直等待缓存区有空间方继续执行。则子进程进入睡眠模式。我们的java 程序也就阻塞在了 process.waitFor();
解决办法
- 所调用的程序中不要有太多的输出到标准出错 、标准输出中。
- Jvm及我们的java程序中,对该子进程的标准输出、标准出错进行读取,从而避免缓存区满。
实现:
方案一
import java.io.InputStreamReader;
import java.io.LineNumberReader;
import java.util.ArrayList;
import java.util.List;
/*
- javac HelloWorld.java
- java HelloWorld
*/
public class HelloWorld {
/**
* @param args
*/
public static void main(String[] args) {
// TODO Auto-generated method stub
System.out.printf("Just test !\n");
String cmd1="/data/dtgy.package/bin/get.rom.list.js.py";
String cmd2="/data/dtgy.package/bin/pack.all.roms.sh";
String cmd="ls -l /data/upload/roms";
List ret;
try {
ret=runShell(cmd1);
ret=runShell(cmd2);
System.out.printf("Just test :"+ret.toString());
//runShell(cmd2);
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
/**
run shell
@param shStr
the shell need to run
@return
-
@throws IOException
*/
public static List runShell(String shStr) throws Exception {
List<String> strList = new ArrayList(); Process process; process = Runtime.getRuntime().exec(new String[]{"/bin/sh","-cI",shStr},null,null); InputStreamReader ir = new InputStreamReader(process .getInputStream()); LineNumberReader input = new LineNumberReader(ir); String line; process.waitFor(); while ((line = input.readLine()) != null){ strList.add(line); } return strList;
}
}
方案二
以后补上
参考链接
http://www.ituring.com.cn/article/details/130
http://siye1982.iteye.com/blog/592405