springboot java ffmpeg 视频压缩、提取视频帧图片、获取视频分辨率

news/2024/12/22 18:12:57/

用到的maven依赖:

lombok依赖就不贴出来了

java">  <dependency><groupId>org.bytedeco</groupId><artifactId>ffmpeg-platform</artifactId><version>4.3.2-1.5.5</version></dependency><dependency><groupId>org.apache.commons</groupId><artifactId>commons-lang3</artifactId><version>3.12.0</version></dependency><dependency><groupId>cn.hutool</groupId><artifactId>hutool-all</artifactId><version>5.5.2</version></dependency>

工具类:

java">import cn.hutool.core.io.IoUtil;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;
import org.bytedeco.ffmpeg.avcodec.AVCodec;
import org.bytedeco.ffmpeg.avcodec.AVCodecContext;
import org.bytedeco.ffmpeg.avformat.AVFormatContext;
import org.bytedeco.ffmpeg.global.avcodec;
import org.bytedeco.ffmpeg.global.avformat;
import org.bytedeco.ffmpeg.global.avutil;
import org.bytedeco.javacpp.Loader;
import org.bytedeco.javacpp.PointerPointer;import java.io.*;
import java.util.concurrent.TimeUnit;@Slf4j
public class VideoUtils {static class LazyFfmpeg {private static final String ffmpeg = Loader.load(org.bytedeco.ffmpeg.ffmpeg.class);}public static String ffmpeg() {return LazyFfmpeg.ffmpeg;}/*** 压缩视频* @param inputFilePath 压缩前视频地址* @param outputFilePath 压缩后视频地址*/public static void compressVideo(String inputFilePath, String outputFilePath) {if (StringUtils.isAnyBlank(inputFilePath, outputFilePath)) {throw new RuntimeException("输入视频路径或输出视频文件路径不能为空");}if (StringUtils.equals(inputFilePath, outputFilePath)) {throw new RuntimeException("outputFilePath不能和inputFilePath相同");}validIsFile(new File(inputFilePath));ProcessBuilder processBuilder = new ProcessBuilder(ffmpeg(),"-y",                     // 自动覆盖输出文件"-i", inputFilePath,      // 输入文件路径"-crf","30","-c:v","h264","-preset", "slow",            // 使用较慢的预设来提高压缩效率
//                "-b:v", "1000",  // 设置视频比特率为 1000 kbps
//                "-vf", String.format("scale=%s:%s", 1920, 1080),
//                "-c:a", "copy",           // 保持音频编码不变"-c:a", "aac",                  // 使用 AAC 音频编码"-b:a", "2k",                 // 设置音频比特率为 128 kbpsoutputFilePath            // 输出文件路径);StringBuilder stringBuilder = new StringBuilder();int exitCode;try {Process process = processBuilder.start();// 捕获错误输出processErrorMsg(process, stringBuilder);// 等待 FFmpeg 进程完成exitCode = process.waitFor();} catch (Throwable e) {throw new RuntimeException(e);}if (exitCode != 0) {throw new RuntimeException(stringBuilder.toString());}}/*** 提取图片** @param videoPath 视频路径* @param second    提取指定时间图片* @param timeout   等待的最长时间* @param unit      参数的时间 timeout 单位* @return 图片*/public static byte[] ffmpegExtractImage(String videoPath, Number second, long timeout, TimeUnit unit) {if (timeout <= 0) {throw new IllegalArgumentException("timeout不能小于等于0");}if (second == null) {second = 0;}if (unit == null) {unit = TimeUnit.MINUTES;}File videoFile = new File(videoPath);validIsFile(videoFile);ProcessBuilder extractBuilder = new ProcessBuilder(ffmpeg(),"-ss", second.toString(),"-i", videoPath,"-f", "image2pipe","-vframes", "1",
//                "-vcodec", "png",//如果觉得照片不清晰,就启用此选项,但是照片会变大"-");try {Process process = extractBuilder.start();try (InputStream inputStream = process.getInputStream()) {byte[] bytes = IoUtil.readBytes(inputStream);boolean result = process.waitFor(timeout, unit);if (!result) {throw new RuntimeException("子进程退出之前已超过等待时间");}return bytes;}} catch (IOException | InterruptedException e) {throw new RuntimeException(e);}}/*** 获取视频分辨率** @param videoFilePath 视频路径*/public static int[] getVideoResolution(String videoFilePath) {validIsFile(new File(videoFilePath));AVFormatContext formatContext = avformat.avformat_alloc_context();AVCodecContext codecContext = avcodec.avcodec_alloc_context3(null);// 打开视频文件if (avformat.avformat_open_input(formatContext, videoFilePath, null, null) != 0) {throw new RuntimeException("无法打开视频文件");}// 获取视频流信息if (avformat.avformat_find_stream_info(formatContext, (PointerPointer) null) < 0) {throw new RuntimeException("无法获取视频流信息");}// 查找视频流int videoStreamIndex = -1;for (int i = 0; i < formatContext.nb_streams(); i++) {if (formatContext.streams(i).codecpar().codec_type() == avutil.AVMEDIA_TYPE_VIDEO) {videoStreamIndex = i;break;}}if (videoStreamIndex == -1) {throw new RuntimeException("视频流未找到");}// 获取视频解码器上下文avcodec.avcodec_parameters_to_context(codecContext, formatContext.streams(videoStreamIndex).codecpar());// 查找解码器AVCodec codec = avcodec.avcodec_find_decoder(codecContext.codec_id());if (codec == null) {throw new RuntimeException("无法找到解码器");}// 打开解码器if (avcodec.avcodec_open2(codecContext, codec, (PointerPointer) null) < 0) {throw new RuntimeException("无法打开解码器");}// 获取视频分辨率int width = codecContext.width();int height = codecContext.height();// 清理资源codecContext.close();return new int[]{width, height};}private static void processErrorMsg(Process process, StringBuilder stringBuilder) {new Thread(() -> {try (BufferedReader reader = new BufferedReader(new InputStreamReader(process.getErrorStream()))) {String line;while ((line = reader.readLine()) != null) {stringBuilder.append(line);}} catch (IOException e) {log.error("打印命令行错误日志出现异常  errMsg:{}", e.getMessage(), e);}}).start();}public static void validIsFile(File file) {validExists(file);if (!file.isFile()) {throw new IllegalArgumentException("不是文件");}}public static void validExists(File file) {if (!file.exists()) {throw new IllegalArgumentException("videoPath不存在");}}
}


http://www.ppmy.cn/news/1557258.html

相关文章

电脑除尘更换cpu和显卡硅脂过程及安装win11系统中遇到的问题

原由 电脑是2022款的显卡TRX3050ti 的Y7000p,在使用过程中发现电脑风扇声音很大&#xff0c;想来也是用很久没有清理过灰尘了&#xff0c;在拆卸视频中发现一个换硅脂的&#xff0c;刚好手头有&#xff0c;想来也没换过&#xff0c;所以就直接换了。但修完后直接打开却出现了无…

HarmonyOS(72)事件拦截处理详解

事件拦截 1、参考资料2、HitTestMode3、onTouchIntercept、onTouch、onClick事件执行顺序3.1、系统默认事件传递顺序3.2、子组件拦截事件1、参考资料 HarmonyOS(71) 自定义事件分发之TouchTestStrategy使用说明HarmonyOS(70) ArkUI 事件分发拦截,事件冲突解决方案HitTestModea…

Windows Server 2019 配置PHP环境(图文教程)

操作系统&#xff1a;Windows Server 2019运行模式&#xff1a;IIS10 fastcgi PHP(安装IIS的时候选择上CGI)软件版本&#xff1a;MySQL 5.7.37 解压版 / PHP 7.4.29 / PHP Manager 1.5.0 / phpMyAdmin 5.1.3 1、MySQL 5.7.37 解压版安装&#xff1a; 为什么我会选择解压版而不…

CodeSurfer 和 Tree-sitter对比

CodeSurfer 和 Tree-sitter 都是代码分析工具&#xff0c;但它们的设计理念、功能和使用场景有很大的不同。要决定在解析一个 C 项目 时选择哪个工具&#xff0c;我们需要全面对比它们的特点、优劣和适用场景。 1. 简介对比 CodeSurfer&#xff1a; 是一个专门用于静态分析代码…

Elasticsearch-DSL高级查询操作

一、禁用元数据和过滤数据 1、禁用元数据_source GET product/_search {"_source": false, "query": {"match_all": {}} }查询结果不显示元数据 禁用之前: {"took" : 0,"timed_out" : false,"_shards" : {&quo…

【Qt】显示类控件:QLabel、QLCDNumber、QProgressBar、QCalendarWidget

目录 QLabel QFrame 例子&#xff1a; textFormat pixmap、scaledContents alignment wordWrap、indent、margin buddy QLCDNumber 例子&#xff1a; QTimer QProgressBar 例子&#xff1a; QCalendarWidget 例子&#xff1a; QLabel 标签控件&#xff0c;用来显示…

故障诊断 | 一个小创新:特征提取+KAN分类

往期精彩内容&#xff1a; Python-凯斯西储大学&#xff08;CWRU&#xff09;轴承数据解读与分类处理 基于FFT CNN - BiGRU-Attention 时域、频域特征注意力融合的轴承故障识别模型-CSDN博客 基于FFT CNN - Transformer 时域、频域特征融合的轴承故障识别模型-CSDN博客 P…

使用CNN模型训练图片识别(键盘,椅子,眼镜,水杯,鼠标)

首先是环境&#xff1a; 我是在Anaconda3中的Jupyter Notebook (tensorflow)中进行训练&#xff0c;环境各位自行安装 数据集&#xff1a; 本次数据集五个类型&#xff08;键盘&#xff0c;椅子&#xff0c;眼镜&#xff0c;水杯&#xff0c;鼠标&#xff09;我收集了每个接近两…