java在执行.bat文件时,经常会出现卡住不动的情况,我们称之为阻塞.
发生阻塞的原因是,在使用Runtime.getRuntime.exec("xx:/xxxx.bat")时,会有outputStream,ErrStream,InputStream三个IO流,如果这三个IO流都有数据的传输操作,那么必须对每一个IO流进行处理,如果有一个IO流漏处理了,那么程序将无限期的等待该IO流被处理,继而程序阻塞.
下面,已java打开安装apache2.2的.bat文件为例,说明一下如
何处理阻塞.
package com.multiable.util;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.concurrent.CountDownLatch;
public class TestApache {
public static void main(String[] args) {
try {
final CountDownLatch threadSignal = new CountDownLatch(2);//初始化countDown
System.out.println("Start ......");
final Process pro = Runtime.getRuntime().exec("cmd /c \"c:\\start.bat\"");
pro.getOutputStream().close();
Runnable errThread = new Runnable() {
@Override
public void run() {
try {
InputStream is = pro.getErrorStream();
InputStreamReader isr = new InputStreamReader(is);
BufferedReader br = new BufferedReader(isr);
String line = "";
StringBuilder sb = new StringBuilder();
while((line = br.readLine()) != null) {
sb.append(line);
}
System.out.println("error : " + sb.toString());
} catch (IOException e) {
e.printStackTrace();
}
threadSignal.countDown();//线程结束时计数器减1
}
};
new Thread(errThread).start();
Runnable inputThread = new Runnable() {
@Override
public void run() {
try {
InputStream is = pro.getInputStream();
InputStreamReader isr = new InputStreamReader(is);
BufferedReader br = new BufferedReader(isr);
String line = "";
StringBuilder sb = new StringBuilder();
while((line = br.readLine()) != null) {
sb.append(line);
}
System.out.println("input : " + sb.toString());
is.close();
isr.close();
br.close();
} catch (IOException e) {
e.printStackTrace();
}
threadSignal.countDown();//线程结束时计数器减1
}
};
new Thread(inputThread).start();
try {
threadSignal.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("=========================");
Runtime.getRuntime().exec("cmd /c net start apache2.2");
System.out.println("cmd /c net start apache2.2");
} catch (IOException e) {
e.printStackTrace();
}
}
}
关于OutputStream,程序用不到,直接在开始的时候就应该close掉.对于errStream & inputStream应该分别用一个线程来读取出IO流中的内容.
注意:必须使用线程,否则依然会阻塞.
final CountDownLatch threadSignal = new CountDownLatch(2);//初始化countDown
这一个栏位是用来让主线程等待两个子线程执行完毕用的.