使用ProcessBuilder执行FFmpeg命令,进程一直处于阻塞状态,一直没有返回执行结果

昨晚我在尝试使用Java的ProcessBuilder开辟一个进程执行FFmpeg相关命令对视频进行HLS切片处理的时候,遇到了一个进程阻塞的问题。

我使用的命令 ffmpeg -i D:\\pythonWorkspace\\bilibili-script\\data\\甚至还没说一句谢谢:懂王称愿自掏腰包付宇航员加班工资_哔哩哔哩_bilibili.mp4 -c:v libx264 -c:a aac -f hls -hls_time 10 -hls_list_size 0 D:\\shuzhiworkspace\\Mindhaven\\output\\output.m3u8 在终端命令行执行的时候是没有问题的,一下子就给我出结果了,但是我使用ProcessBuilder开辟的进程,等了半天一直出不来结果,进程陷入了阻塞状态。

问题代码

java 复制代码
@Test  
void test16(){  
  String inputPath = "D:\\pythonWorkspace\\bilibili-script\\data\\甚至还没说一句谢谢:懂王称愿自掏腰包付宇航员加班工资_哔哩哔哩_bilibili.mp4";  
    String outputFolder = "/output";  
    //ffmpeg -i input.mp4 -c:v libx264 -c:a aac -f hls -hls_time 10 -hls_list_size 0 output.m3u8  
    String commend = "ffmpeg -i D:\\pythonWorkspace\\bilibili-script\\data\\甚至还没说一句谢谢:懂王称愿自掏腰包付宇航员加班工资_哔哩哔哩_bilibili.mp4 -c:v libx264 -c:a aac -f hls -hls_time 10 -hls_list_size 0 D:\\shuzhiworkspace\\Mindhaven\\output\\output.m3u8";  
    //构建FFmpeg命令参数  
    String[] cmd = {  
            "ffmpeg",  
            "-i",inputPath,  
            "-c:v", "libx264",  
            "-c:a", "aac",  
            "-f", "hls",  
            "-hls_time", "10",  
            "-hls_list_size", "0",  
            outputFolder+"/output.m3u8"  
    };  
    //检查目录是否存在  
    File outputDir = new File(outputFolder);  
    if(!outputDir.exists()){  
        boolean mkdirs = outputDir.mkdirs();  
        if(!mkdirs){  
            System.out.println("创建目录失败");  
            return;  
        }  
    }  
    ProcessBuilder processBuilder = new ProcessBuilder(cmd);  
    System.out.println("工作路径:"+processBuilder.directory());  
  
    try {  
        System.out.println("开始执行FFmpeg命令...");  
        long l = System.currentTimeMillis();  
        Process process = processBuilder.start();  
        System.out.println("pid:"+process.pid());  
        int exitCode = process.waitFor();  
        if(exitCode == 0){  
            System.out.println("视频切片成功!");  
        }  
        System.out.println("FFmpeg命令执行完成,耗时:"+(System.currentTimeMillis()-l)+"ms");  
    } catch (IOException | InterruptedException e) {  
        e.printStackTrace();  
    }}

问题解决方案

今天早上,我询问了一下AI大模型,在它的思维链中发现了问题所在。原来FFmpeg在执行转码操作的时候,会持续的输出日志信息到 stdoutstderr这两个流中,而流数据又会先输入到缓冲区中。如果父进程(Java进程)没有主动读取这些流,当缓冲区满了的时候,子进程就会暂停执行并等待父进程(Java进程)消费数据,然而我的Java程序并没有做消费数据的处理,因而形成了死锁。从而表现在我的 int exitCode = process.waitFor(); 无限卡住,无法返回退出码。

而我在终端命令行直接执行时可以成功很快的返回之处在于,终端命令行的进程回去消费stdoutstderr 中的日志信息并输出在终端上,也就是下图所示的输出。

因此,问题的解决方法就是在Java程序中再开两个子线程,去读取消费stdoutstderr 中的日志信息,避免缓存区阻塞,这样我们的问题就基本解决了。

同时,我们还可以waitFor()方法设置一个超时中断机制,避免因其他未知的原因导致进程无限阻塞。

问题解决后的代码

java 复制代码
@Test  
void test17(){  
    String inputPath = "D:\\pythonWorkspace\\bilibili-script\\data\\甚至还没说一句谢谢:懂王称愿自掏腰包付宇航员加班工资_哔哩哔哩_bilibili.mp4";  
    String outputFolder = "/output";  
    //ffmpeg -i input.mp4 -c:v libx264 -c:a aac -f hls -hls_time 10 -hls_list_size 0 output.m3u8  
    String commend = "ffmpeg -i D:\\pythonWorkspace\\bilibili-script\\data\\甚至还没说一句谢谢:懂王称愿自掏腰包付宇航员加班工资_哔哩哔哩_bilibili.mp4 -c:v libx264 -c:a aac -f hls -hls_time 10 -hls_list_size 0 D:\\shuzhiworkspace\\Mindhaven\\output\\output.m3u8";  
    //构建FFmpeg命令参数  
    String[] cmd = {  
            "ffmpeg",  
            "-i",inputPath,  
            "-c:v", "libx264",  
            "-c:a", "aac",  
            "-f", "hls",  
            "-hls_time", "10",  
            "-hls_list_size", "0",  
            outputFolder+"/output.m3u8"  
    };  
    //检查目录是否存在  
    File outputDir = new File(outputFolder);  
    if(!outputDir.exists()){  
        boolean mkdirs = outputDir.mkdirs();  
        if(!mkdirs){  
            System.out.println("创建目录失败");  
            return;  
        }  
    }  
    ProcessBuilder processBuilder = new ProcessBuilder(cmd);  
    System.out.println("工作路径:"+processBuilder.directory());  
  
    try {  
        System.out.println("开始执行FFmpeg命令...");  
        long l = System.currentTimeMillis();  
        Process process = processBuilder.start();  
        // 启动线程读取 stdout        Thread stdoutThread = new Thread(() -> {  
            try (BufferedReader reader = new BufferedReader(  
                    new InputStreamReader(process.getInputStream()))) {  
                String line;  
                while ((line = reader.readLine()) != null) {  
                    System.out.println("[FFmpeg] " + line);  
                }  
            } catch (IOException e) { e.printStackTrace(); }  
        });  
  
        // 启动线程读取 stderr(关键!FFmpeg 错误信息在此流)  
        Thread stderrThread = new Thread(() -> {  
            try (BufferedReader reader = new BufferedReader(  
                    new InputStreamReader(process.getErrorStream()))) {  
                String line;  
                while ((line = reader.readLine()) != null) {  
                    System.err.println("[FFmpeg-Error] " + line);  
                }  
            } catch (IOException e) { e.printStackTrace(); }  
        });  
  
        stdoutThread.start();  
        stderrThread.start();  
        System.out.println("pid:"+process.pid());  
        boolean exitCode = process.waitFor(60l,TimeUnit.SECONDS);  
        if(exitCode){  
            System.out.println("视频切片成功!");  
        }  
        System.out.println("FFmpeg命令执行完成,耗时:"+(System.currentTimeMillis()-l)+"ms");  
    } catch (IOException | InterruptedException e) {  
        e.printStackTrace();  
    }}

至此,使用ProcessBuilder执行FFmpeg命令,进程一直处于阻塞状态,一直没有返回执行结果的这个问题就解决了。拜拜~

相关推荐
zzywxc78719 分钟前
大模型落地实践指南:从技术路径到企业级解决方案
java·人工智能·python·microsoft·golang·prompt
Industio_触觉智能22 分钟前
瑞芯微RK35XX系列FFmpeg硬件编解码实测,详细性能对比!
ffmpeg·rk3588·rk3568·编解码·rk3562·rk3576
相与还24 分钟前
IDEA+SpringBoot实现远程DEBUG到本机
java·spring boot·intellij-idea
小杨勇敢飞26 分钟前
IDEA 2024 中创建 Maven 项目的详细步骤
java·ide·intellij-idea
小狮子安度因30 分钟前
FFmpeg暂停、逐帧和音量
ffmpeg
洛可可白1 小时前
把 Vue2 项目“黑盒”嵌进 Vue3:qiankun 微前端实战笔记
前端·vue.js·笔记
野犬寒鸦1 小时前
从零起步学习Redis || 第四章:Cache Aside Pattern(旁路缓存模式)以及优化策略
java·数据库·redis·后端·spring·缓存
白水先森1 小时前
C语言作用域与数组详解
java·数据结构·算法
想唱rap1 小时前
直接选择排序、堆排序、冒泡排序
c语言·数据结构·笔记·算法·新浪微博
小狮子安度因2 小时前
FFmpeg过滤器实战:水印处理
ffmpeg·myeclipse