- 1.基本使用
- 2.使用时的注意事项
- 3.调用waitFor()阻塞主线程
- 3.1.问题描述
- 3.2.解决方案
// 需要执行的指令 String commandStr = "xxx"; Runtime run = Runtime.getRuntime(); run.exec(commandStr)
这段代码的意思是新开一个子线程,在子线程中执行commandStr指令
2.使用时的注意事项如果输入的指令属于某个应用程序(如ffmpeg、MySQL)
在指令的前面必须要加上该应用程序所在的绝对路径。
比如下方指令可以在XShell中正常执行
# 下方命令可以被正常执行 ffmpeg -i /video/1.mp3 -ss 00:00:00.001 -vframs 1 /video/picture/1.jpg
而如果在java中通过RunTime执行,则需要加上ffmpeg所在的绝对路径
比如我的ffmpeg在/usr/local/ffmpeg/bin目录下:
// java中,commandStr前需要加上应用程序的绝对路径path
// path = "/usr/local/ffmpeg/bin";
String path = File.separator +
"usr" +
File.separator +
"local" +
File.separator +
"ffmpeg" +
File.separator +
"bin";
// commandStr = "/usr/local/ffmpeg/bin/ffmpeg -i /video/1.mp3 -ss 00:00:00.001 -vframs 1 /video/picture/1.jpg";
String commandStr = path +
File.separator +
"ffmpeg -i " +
File.separator +
"video" +
File.separator +
"1.mp3 -ss 00:00:00.001 -vframs 1 " +
File.separator +
"video" +
File.separator +
"picture" +
File.separator +
"1.jpg";
Runtime run = Runtime.getRuntime();
run.exec(commandStr)
3.调用waitFor()阻塞主线程
3.1.问题描述
由于Runtime.exec()是新开一个子线程去完成的,很难保证java在执行下一步操作前,子线程的指令执行完毕。我们假设有这样的场景:
目的:通过指令将xxx目录下的文件A重命名为B,并在java端读取B
那么此时代码如下:
// 创建子线程执行指令
String commandStr = "mv /xxx/A /xxx/B";
Runtime run = Runtime.getRuntime();
run.exec(commandStr);
// 尝试读取文件B
File file = new File("/xxx/B");
Runtime会开启一个子线程,此时很可能出现这种情况:第4行代码尚未开始执行,java便已经开始自行第7行代码了,而此时就会抛出NPE。
3.2.解决方案调用waitFor()可以让子线程在执行指令时阻塞主线程,这样就能够确保在指令执行完毕之前,不会继续往下执行:
// 创建子线程执行指令
String commandStr = "mv /xxx/A /xxx/B";
Runtime run = Runtime.getRuntime();
// 阻塞主线程直到子线程执行完毕
run.exec(commandStr).waitFor();
// 读取文件B
File file = new File("/xxx/B");



