1.Java调用命令行,如果没有额外环境变量,不指定工作路径,Runtime有两个方法
public Process exec(String command) public Process exec(String cmdarray[])
ffmpeg推流本地视频命令如下
ffmpeg -re -i test.mp4 <param> <url>
ffmpeg -re -i test.mp4 <param> "<url> socks=***" // 代理
如果用exec(String command) 有代理的情况下,就算将双引号包含进去,也会不成功,会将<url>部分当做输入的视频文件位置。建议使用exec(String cmdarray[])方法。
在使用exec(String cmdarray[])时,内部处理是当如果中间有空格时,在首尾加上双引号,具体使用具体分析。
2.调用命令行后,程序可能正常推流推一会儿,然后程序block住,后台进程ffmpeg还没死。
Runtime.exec()执行时JVM会产生一个子进程,该进程与JVM建立三个通道链接:标准输入,标准输出,标准错误。
Java本地的系统对标准输入和输出所提供的缓冲池有限,所以错误的对标准输出快速的写入和从标准输入快速的读入都有可能造成子进程死锁。
子进程的输出流,也就是JVM的输入流。子进程不断向控制台输出,如果Java没有把输入流及时清空,会导致缓存区满,导致死锁。
解决办法就是及时清空输入流,开两个线程把process.getInputStream()和process.getErrorStream()读出来就可以。对于ffmpeg只需要process.getErrorStream()读出来就可以了。
Process process = Runtime.getRuntime().exec(command);
BufferedReader br= new BufferedReader(new InputStreamReader(process.getErrorStream())); while ((line = br.readLine()) != null) { System.out.println(line); }
process.waitFor();
或者ProcessBuilder重定向标准错误
ProcessBuilder processBuilder = new ProcessBuilder(commands); processBuilder.redirectErrorStream(true); Process process = processBuilder.start(); ProcessContext.getInstance().addProcess(process); BufferedReader br = new BufferedReader(new InputStreamReader(process.getInputStream())); String line = null; while ((line = br.readLine()) != null) { System.out.println(line); }
process.waitFor();
3.在虚拟机上测试时,模拟服务器瞬间断网的情况(拔网线),在虚拟机上跑的程序卡住了。
这种情况在正常服务器上没有问题,只在自己的虚拟机上有问题,原因不明,建议这种case在物理机上测试。
4.ffmpeg推本地视频文件时占用cpu过高,每个进程大约78-80%。
cpu主要是被图像转码占用了。如果是本地视频并且不需要对图像转码的情况,使用参数 -vcodec copy不进行转码只是复制。会大大减低cpu使用率(仅占用2%左右)。
如果就是有转码的需求,可以尝试参数 -threads 2。
帮助:
1.ffmepg推流的返回值(process.exitValue()),0为成功,其他失败。
2.使用process.waitFor()等待子进程。
3.-ss **:**:** 跳过指定时长,对参数位置有要求。