Java 执行系统命令
在Java中执行系统命令,主要是使用ProcessBuilder和Runtime.getRuntime().exec()。而在这里主要是介绍两种方法的使用。
使用情景是在linux系统中,使用mencoder来进行视频转码。将视频转为flv格式,因为转为flv格式命令较为简单,如要转为MP4格式,可以看官方文档介绍
一.使用Runtime
##上传之后的视频文件名为test.tmpmedia
String command = "mencoder test.tmpmedia -o test.flv -ofps 25 -of lavf -oac mp3lame -srate 22050 -ovc lavc -lavcopts "
+ "vcodec=flv:keyint=59:vbitrate=6000:mbd=2:trell:v4mv:o=mpv_flagg=cbp_rd:last_pred=3 -vf harddup,scale=480:256";
Process p = Runtime.getRuntime().exec(command);
##读取命令的输出信息
InputStream is = p.getInputStream();
BufferedReader reader = new BufferedReader(new InputStreamReader(is));
p.waitFor();
if (p.exitValue() != 0) {
//说明命令执行失败
//可以进入到错误处理步骤中
}
##打印输出信息
String s = null;
while ((s = reader.readLine()) != null) {
System.out.println(s);
}
其中waitFor()
方法会阻塞当前进程,直到命令执行结束。而exitValue
不会阻塞进程,但是调用exitValue
的时候,如果命令没有执行完成就会报错,感觉这样设计挺奇怪的。
所以,想要获取到命令是否执行成功,就搭配waitFor
和exitValue
。
使用Runtime执行命令感觉是比较方便的,直接可以将命令写入exec()
中。如果仅仅是这样写的话,转码是会一直卡死在那里的。而你将命令直接放到终端中执行,又不会出现问题。
原因就在转码时候,会输出大量的信息(标准输出和标准错误),如果不清理java的缓存区,就会导致缓存区满而命令无法继续执行的情况。
那么解决办法肯定就是清理缓存区,而使用Runtime清理的话,你至少得再开另外一个线程,才能同时getInputStream
和getErrorStream
,这样并不是我喜欢的,所以便采用了ProcessBuilder
二.使用ProcessBuilder
#将命令分解为List存储
String[] commandSplit = command.split(" ");
List<String> lcommand = new ArrayList<String>();
for (int i = 0; i < commandSplit.length; i++) {
lcommand.add(commandSplit[i]);
}
ProcessBuilder processBuilder = new ProcessBuilder(lcommand);
processBuilder.redirectErrorStream(true);
Process p = processBuilder.start();
InputStream is = p.getInputStream();
BufferedReader bs = new BufferedReader(new InputStreamReader(is));
p.waitFor();
if (p.exitValue() != 0) {
//说明命令执行失败
//可以进入到错误处理步骤中
}
String line = null;
while ((line = bs.readLine()) != null) {
System.out.println(line);
resultLog += line;
}
ProcessBuilder可以使用redirectErrorStream
将标准输出和标准错误流合并,然后使用getInputStream
获取到流,放入BufferedReader中打印出来,便可解决缓存区满的问题
唯一感觉不太方便就是命令执行已List<String>
或者String []
的形式传入。
上面的两种方法都是以阻塞的方式执行命令的,要想在后台执行命令,并且命令执行成功后通知,失败后记录日志,那就只能在主线程中开一个线程,专门用来转码。可以参考下一篇文章《
Spring中多线程转码,bean的注入问题》