用Java的Process执行命令行,ffmpeg抽帧到一千多帧图片卡住,不报错。

项目场景:

最近需要写一个抽帧、推流的工具类,抽帧写好测试的时候也没问题,等到真正用的时候就发现各种问题。


问题描述

用Java执行ffmpeg抽帧命令,测试的时候没有问题,后来发现抽帧图片多了就会卡住。

刚开始觉得可能是网速的问题,测试了一下也没问题阿,抽帧图片多了才发现不知道怎么回事不抽了,最关键的是进程都还没掉。


原因分析:

刚开始觉得可能是这个命令的问题,就直接拿命令在Linux执行发现没这个问题,在Java执行才会有这个问题,然后想着直接用shell脚本试试,也是没有卡住的情况,既然这样,就想着拿Java执行shell脚本再执行命令行,结果还是出现了一样的情况,这是最想不通的一点,这都不是一个进程了,shell脚本后台运行,按理来说就算Java调用shell脚本的进程直接杀掉也还会执行,但是还是会到了一千多张就卡住了。后来就尝试用Javacv 呗,刚开始尝试的还不如直接调用process,只能抽九百多张,这个因为没情况缓冲区,加了

java 复制代码
grabber.setVideoOption("fflags", "nobuffer");

以后可以抽到两千多张,还是到不了想要的效果。

解决方案:

想了想会不会process也是这样的情况,当时process只清空了输入流和错误流,没有对输出流进行处理。添加如下代码

java 复制代码
  ProcessBuilder processBuilder = new ProcessBuilder("cmd",path);
  processBuilder.redirectErrorStream(true);
  try {
  Process process = null;
  process = processBuilder.start();
              // 读取标准输出和错误
            InputStream inputStream = process.getInputStream();
            BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));
            String line;
            while ((line = reader.readLine()) != null) {
                log.info(line); // 处理标准输出和错误
            }
        } catch (Exception e) {
            throw new RuntimeException(e);
        }

这样才给问题解决了,如果需要给前端返回进程号,可以使用下面的代码(jdk1.8),jdk版本高的可以直process.getPid();异步执行process.waitFor(); 提前返回进程号。

java 复制代码
 pidFiled = process.getClass().getDeclaredField("pid");
 pidFiled.setAccessible(true);
 String pid = String.valueOf(pidFiled.getInt(process));

根据进程号结束抽帧的代码:

java 复制代码
/**
 * @param pid 进程号
 * @return 0表示正常
 */
public Integer stopByPid(String pid) {
    if (!StringUtils.hasText(pid)) {
        throw new BadRequestException("Pid cannot be empty");
    }
    String command = "kill -9 " + pid;
    log.info(command);
    Integer exitCode = 0;
    ProcessBuilder processBuilder = new ProcessBuilder();
    processBuilder.command("bash", "-c", command);
    try {
        Process process = processBuilder.start();
        BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream()));
        String line;
        while ((line = reader.readLine()) != null) {
            log.error(line);
        }
        exitCode = process.waitFor();
        log.info("Exited with error code : " + exitCode);
    } catch (Exception e) {
        e.printStackTrace();
    }
    return exitCode;
}
相关推荐
快乐就好ya29 分钟前
Java多线程
java·开发语言
IT学长编程34 分钟前
计算机毕业设计 二手图书交易系统的设计与实现 Java实战项目 附源码+文档+视频讲解
java·spring boot·毕业设计·课程设计·毕业论文·计算机毕业设计选题·二手图书交易系统
CS_GaoMing1 小时前
Centos7 JDK 多版本管理与 Maven 构建问题和注意!
java·开发语言·maven·centos7·java多版本
艾伦~耶格尔1 小时前
Spring Boot 三层架构开发模式入门
java·spring boot·后端·架构·三层架构
man20172 小时前
基于spring boot的篮球论坛系统
java·spring boot·后端
2401_858120532 小时前
Spring Boot框架下的大学生就业招聘平台
java·开发语言
S hh2 小时前
【Linux】进程地址空间
java·linux·运维·服务器·学习
Java探秘者2 小时前
Maven下载、安装与环境配置详解:从零开始搭建高效Java开发环境
java·开发语言·数据库·spring boot·spring cloud·maven·idea
攸攸太上2 小时前
Spring Gateway学习
java·后端·学习·spring·微服务·gateway
2301_786964362 小时前
3、练习常用的HBase Shell命令+HBase 常用的Java API 及应用实例
java·大数据·数据库·分布式·hbase