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

相关推荐
摇滚侠9 分钟前
MyBatis 入门到项目实战 特殊 SQL 的执行 34-37
java·sql·mybatis
万物更新_29 分钟前
vue框架
前端·javascript·vue.js·笔记
phltxy39 分钟前
Spring AI 可观测性与 Zipkin 实战
java·人工智能·spring
兰令水1 小时前
leecodecode【面试150】【2026.6.13打卡-java版本】
java·算法·leetcode
.道阻且长.1 小时前
C++ string 操作指南:接口解析
java·c语言·开发语言·c++
蚰蜒螟1 小时前
Java 对象的内存密语:从字段偏移量计算到 Unsafe 访问的完整链路
java·开发语言
IT 行者1 小时前
GitHub Spec Kit 实战(六):/speckit.implement 怎么用、怎么审、怎么发现 spec 阶段的遗漏——五部曲收官
java·驱动开发·github·ai编程·claude
上海观智网络1 小时前
上海小程序定制开发合同怎么签?需要注意什么?
经验分享·笔记·小程序
星辰_mya1 小时前
CountDownLatch深度解析
java·开发语言·后端·架构
伊甸31 小时前
从企业级项目学敏感词过滤:DFA算法与双层缓存实战
java·算法·缓存