使用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命令,进程一直处于阻塞状态,一直没有返回执行结果的这个问题就解决了。拜拜~

相关推荐
Long_poem7 分钟前
【自学笔记】PHP语言基础知识点总览-持续更新
android·笔记·php
程序猿大波12 分钟前
基于Java,SpringBoot,Vue,HTML高校社团信息管理系统设计
java·vue.js·spring boot
小李同学_LHY43 分钟前
微服务架构中的精妙设计:环境和工程搭建
java·spring·微服务·springcloud
慕容魏43 分钟前
面经分享,中科创达(安卓开发,二面挂)
java·开发语言
不辉放弃1 小时前
Java/Scala是什么
java·scala
喵手1 小时前
Java实现视频格式转换的完整指南:从FFmpeg到纯Java方案!
java·开发语言·ffmpeg
钰爱&1 小时前
【Qt】ffmpeg编码—存储(H264)
c++·qt·ffmpeg
天上掉下来个程小白2 小时前
Redis-04.Redis常用命令-字符串常用命令
java·数据库·redis·springboot·苍穹外卖
yunteng5212 小时前
音视频(二)ffmpeg编译及推流
ffmpeg·音视频·h264·媒体推流
Zz_waiting.2 小时前
多线程 - 线程安全 2 -- > 死锁问题
java·开发语言