FFmpeg获取视频详情

话不多说,直接上代码:

pom依赖:

        <!--视频多媒体工具包 包含 FFmpeg、OpenCV--><dependency><groupId>org.bytedeco</groupId><artifactId>javacv-platform</artifactId><version>1.5.3</version></dependency><!-- https://mvnrepository.com/artifact/commons-lang/commons-lang --><dependency><groupId>commons-lang</groupId><artifactId>commons-lang</artifactId><version>2.6</version></dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId></dependency><dependency><groupId>org.openjfx</groupId><artifactId>javafx-controls</artifactId><version>16</version></dependency>

工具类:

package com.example.mybatisdemo02.util;import com.example.mybatisdemo02.model.VideoInfoVO;
import javafx.application.Application;
import javafx.geometry.Pos;
import javafx.scene.Scene;
import javafx.scene.control.Slider;
import javafx.scene.layout.BorderPane;
import javafx.scene.layout.HBox;
import javafx.scene.media.Media;
import javafx.scene.media.MediaPlayer;
import javafx.scene.media.MediaView;
import javafx.stage.Stage;
import org.apache.commons.lang.StringUtils;
import org.bytedeco.javacv.FFmpegFrameGrabber;
import org.bytedeco.javacv.Frame;
import org.bytedeco.javacv.FrameGrabber;
import org.bytedeco.javacv.Java2DFrameConverter;
import javafx.scene.control.Button;
import javax.imageio.ImageIO;
import javax.imageio.ImageReader;
import javax.imageio.stream.ImageInputStream;
import javax.swing.*;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.*;
import java.math.BigDecimal;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
import java.nio.channels.FileChannel;
import java.util.UUID;public class FFmpegUtil {private static final String IMAGE_SUFFIX = "png";/*** 获取视频时长,单位为秒S* @return*/public static long getVideoTime( String localPath) throws FrameGrabber.Exception {
//        String localPath = "C:/Users/Administrator/Desktop/dab2d14cad0244229e228e7bf297dd9a.flv";FFmpegFrameGrabber grabber = FFmpegFrameGrabber.createDefault(localPath);grabber.start();long duration = grabber.getLengthInTime() / (1000 * 1000);grabber.stop();System.out.println("此视频时长(s/秒):"+duration);return duration;}/*** 获取视频帧图片* @param file 视频源* @param number 第几帧* @param dir 文件存放根目录* @param args 文件存放根目录* @return*/@SuppressWarnings("resource")public static String videoImage(File file, Integer number, String dir, String args) {String picPath = StringUtils.EMPTY;FFmpegFrameGrabber ff = new FFmpegFrameGrabber(file);try {ff.start();int i = 0;int length = ff.getLengthInFrames();Frame frame = null;while (i < length) {frame = ff.grabFrame();//截取第几帧图片if ((i > number) && (frame.image != null)) {//获取生成图片的路径picPath = getImagePath(args);//执行截图并放入指定位置doExecuteFrame(frame, dir + File.separator + picPath);break;}i++;}ff.stop();} catch (FrameGrabber.Exception e) {e.printStackTrace();}return picPath;}/*** 截取缩略图* @param frame* @param targerFilePath 图片存放路径*/public static void doExecuteFrame(Frame frame, String targerFilePath) {//截取的图片if (null == frame || null == frame.image) {return;}Java2DFrameConverter converter = new Java2DFrameConverter();BufferedImage srcImage = converter.getBufferedImage(frame);int srcImageWidth = srcImage.getWidth();int srcImageHeight = srcImage.getHeight();//对帧图片进行等比例缩放(缩略图)int width = 480;int height = (int) (((double) width / srcImageWidth) * srcImageHeight);BufferedImage thumbnailImage = new BufferedImage(width, height, BufferedImage.TYPE_3BYTE_BGR);thumbnailImage.getGraphics().drawImage(srcImage.getScaledInstance(width, height, Image.SCALE_SMOOTH), 0, 0, null);File output = new File(targerFilePath);try {ImageIO.write(thumbnailImage, IMAGE_SUFFIX, output);} catch (IOException e) {e.printStackTrace();}}/*** 生成图片的相对路径* @param args 传入生成图片的名称、为空则用UUID命名* @return 例如 upload/images.png*/public static String getImagePath(String args) {if (StringUtils.isNotEmpty(args)) {return args + "." + IMAGE_SUFFIX;}return getUUID() + "." + IMAGE_SUFFIX;}/*** 生成唯一的uuid* @return uuid*/public static String getUUID(){return UUID.randomUUID().toString().replace("-","");}/*** 时长格式换算* @param duration 时长* @return HH:mm:ss*/public static String formatDuration(Long duration) {String formatTime = StringUtils.EMPTY;double time = Double.valueOf(duration);if (time > -1) {int hour = (int) Math.floor(time / 3600);int minute = (int) (Math.floor(time / 60) % 60);int second = (int) (time % 60);if (hour < 10) {formatTime = "0";}formatTime += hour + ":";if (minute < 10) {formatTime += "0";}formatTime += minute + ":";if (second < 10) {formatTime += "0";}formatTime += second;}return formatTime;}/*** 获取图片大小Kb* @param urlPath* @return*/public static String getImageSize(String urlPath){// 得到数据byte[] imageFromURL = getImageFromURL(urlPath);// 转换String byte2kb = bytes2kb(imageFromURL.length);return byte2kb;}/*** 根据图片地址获取图片信息** @param urlPath 网络图片地址* @return*/public static byte[] getImageFromURL(String urlPath) {// 字节数组byte[] data = null;// 输入流InputStream is = null;// Http连接对象HttpURLConnection conn = null;try {// Url对象URL url = new URL(urlPath);// 打开连接conn = (HttpURLConnection) url.openConnection();// 打开读取 写入是setDoOutput
//	        conn.setDoOutput(true);conn.setDoInput(true);// 设置请求方式conn.setRequestMethod("GET");// 设置超时时间conn.setConnectTimeout(6000);// 得到访问的数据流is = conn.getInputStream();// 验证访问状态是否是200 正常if (conn.getResponseCode() == 200) {data = readInputStream(is);} else {data = null;}} catch (MalformedURLException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally {try {if (is != null) {// 关闭流is.close();}} catch (IOException e) {e.printStackTrace();}// 关闭连接conn.disconnect();}return data;}/*** 将流转换为字节** @param is* @return*/public static byte[] readInputStream(InputStream is) {/*** 捕获内存缓冲区的数据,转换成字节数组* ByteArrayOutputStream类是在创建它的实例时,程序内部创建一个byte型别数组的缓冲区,然后利用ByteArrayOutputStream和ByteArrayInputStream的实例向数组中写入或读出byte型数据。* 在网络传输中我们往往要传输很多变量,我们可以利用ByteArrayOutputStream把所有的变量收集到一起,然后一次性把数据发送出去。*/ByteArrayOutputStream baos = new ByteArrayOutputStream();// 创建字节数组 1024 = 1Mbyte[] buffer = new byte[1024];// 防止无限循环int length = -1;try {// 循环写入数据到字节数组while ((length = is.read(buffer)) != -1) {baos.write(buffer, 0, length);}// 强制刷新,扫尾工作,主要是为了,让数据流在管道的传输工程中全部传输过去,防止丢失数据baos.flush();} catch (IOException e) {e.printStackTrace();}// 字节流转换字节数组byte[] data = baos.toByteArray();try {// 关闭读取流is.close();// 关闭写入流baos.close();} catch (IOException e) {e.printStackTrace();}return data;}/*** 获取本地图片的字节数** @param imgPath* @return*/public static String pathSize(String imgPath) {File file = new File(imgPath);FileInputStream fis;int fileLen = 0;try {fis = new FileInputStream(file);fileLen = fis.available();} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();}return bytes2kb(fileLen);}/*** 将获取到的字节数转换为KB,MB模式** @param bytes* @return*/public static String bytes2kb(long bytes) {BigDecimal filesize = new BigDecimal(bytes);
//        BigDecimal megabyte = new BigDecimal(1024 * 1024);
//        float returnValue = filesize.divide(megabyte, 2, BigDecimal.ROUND_UP).floatValue();
//        if (returnValue > 1)
//            return (returnValue + "MB");
//        BigDecimal kilobyte = new BigDecimal(1024);
//        returnValue = filesize.divide(kilobyte, 2, BigDecimal.ROUND_UP).floatValue();
//        return (returnValue + "KB");return filesize.toString();}public static String getImageFormat(String imagePath) throws IOException {// 字节数组byte[] data = null;String format = null;// 输入流InputStream is = null;// Http连接对象HttpURLConnection conn = null;ImageInputStream imageInputStream = null;try {// Url对象URL url = new URL(imagePath);// 打开连接conn = (HttpURLConnection) url.openConnection();// 打开读取 写入是setDoOutput
//	        conn.setDoOutput(true);conn.setDoInput(true);// 设置请求方式conn.setRequestMethod("GET");// 设置超时时间conn.setConnectTimeout(6000);// 得到访问的数据流is = conn.getInputStream();// 验证访问状态是否是200 正常if (conn.getResponseCode() == 200) {imageInputStream = ImageIO.createImageInputStream(is);ImageReader imageReader = ImageIO.getImageReaders(imageInputStream).next();format = imageReader.getFormatName();} else {format = null;}} catch (MalformedURLException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally {try {if (is != null) {// 关闭流is.close();}if(imageInputStream != null){imageInputStream.close();}} catch (IOException e) {e.printStackTrace();}// 关闭连接conn.disconnect();}return format;}/*** 将inputStream转化为file* @param is* @param file 要输出的文件目录*/public static void inputStream2File(InputStream is, File file) throws IOException {OutputStream os = null;try {os = new FileOutputStream(file);int len = 0;byte[] buffer = new byte[8192];while ((len = is.read(buffer)) != -1) {os.write(buffer, 0, len);}} finally {os.close();is.close();}}/*** 获取视频详情* @param file* @return*/public static VideoInfoVO getVideoInfo(String file) {VideoInfoVO videoInfoVO = new VideoInfoVO();FFmpegFrameGrabber grabber = null;try {grabber = FFmpegFrameGrabber.createDefault(file);// 启动 FFmpeggrabber.start();// 读取视频帧数videoInfoVO.setLengthInFrames(grabber.getLengthInVideoFrames());// 读取视频帧率videoInfoVO.setFrameRate(grabber.getVideoFrameRate());// 读取视频秒数videoInfoVO.setDuration(grabber.getLengthInTime() / 1000000.00);// 读取视频宽度videoInfoVO.setWidth(grabber.getImageWidth());// 读取视频高度videoInfoVO.setHeight(grabber.getImageHeight());videoInfoVO.setAudioChannel(grabber.getAudioChannels());videoInfoVO.setVideoCode(grabber.getVideoCodecName());videoInfoVO.setAudioCode(grabber.getAudioCodecName());// String md5 = MD5Util.getMD5ByInputStream(new FileInputStream(file));videoInfoVO.setSampleRate(grabber.getSampleRate());return videoInfoVO;} catch (Exception e) {e.printStackTrace();return null;} finally {try {if (grabber != null) {// 此处代码非常重要,如果没有,可能造成 FFmpeg 无法关闭grabber.stop();grabber.release();}} catch (FFmpegFrameGrabber.Exception e) {e.printStackTrace();}}}/*** 截取视频获得指定帧的图片(截取视频帧封面)** @param video   源视频文件*/public void getVideoPic(InputStream video) {FFmpegFrameGrabber ff = new FFmpegFrameGrabber(video);try {ff.start();// 截取中间帧图片(具体依实际情况而定)int i = 0;int length = ff.getLengthInFrames();
//            int middleFrame = length / 2;int middleFrame = 5;Frame frame = null;while (i < length) {frame = ff.grabFrame();if ((i > middleFrame) && (frame.image != null)) {break;}i++;}// 截取的帧图片Java2DFrameConverter converter = new Java2DFrameConverter();BufferedImage srcImage = converter.getBufferedImage(frame);int srcImageWidth = srcImage.getWidth();int srcImageHeight = srcImage.getHeight();BufferedImage image = new BufferedImage(srcImageWidth, srcImageHeight, BufferedImage.TYPE_INT_ARGB);Graphics2D g2d = image.createGraphics();g2d.setColor(Color.RED);g2d.fillRect(0, 0, srcImageWidth, srcImageHeight);g2d.setColor(Color.WHITE);g2d.drawString("Hello, World!", 50, 100);g2d.dispose();File file = new File("image.png");try {ImageIO.write(image, "png", file);} catch (IOException e) {e.printStackTrace();}ff.stop();} catch (java.lang.Exception e) {e.printStackTrace();System.out.println(e);}}/*** 获取文件大小* @param inputStream* @return*/public String getFileSize(InputStream inputStream){FileChannel fc = null;String size = "0";try {FileInputStream fis = convertToFileInputStream(inputStream);fc = fis.getChannel();BigDecimal fileSize = new BigDecimal(fc.size());size = String.valueOf(fileSize);} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally {if (null != fc) {try {fc.close();} catch (IOException e) {e.printStackTrace();}}}return size;}/*** inputStream转FileInputStream* @param inputStream* @return* @throws IOException*/public static FileInputStream convertToFileInputStream(InputStream inputStream) throws IOException {File tempFile = File.createTempFile("temp", ".tmp");tempFile.deleteOnExit();try (FileOutputStream outputStream = new FileOutputStream(tempFile)) {byte[] buffer = new byte[1024];int bytesRead;while ((bytesRead = inputStream.read(buffer)) != -1) {outputStream.write(buffer, 0, bytesRead);}}return new FileInputStream(tempFile);}public void start(Stage primaryStage) {String videoPath = "C:\\Users\\DELL\\Desktop\\test.mp4";double position = 1200.0; // 播放位置(以秒为单位)// 创建媒体播放器Media media = new Media(new File(videoPath).toURI().toString());MediaPlayer mediaPlayer = new MediaPlayer(media);mediaPlayer.setOnReady(() -> {mediaPlayer.seek(mediaPlayer.getTotalDuration().multiply(position / mediaPlayer.getTotalDuration().toSeconds()));mediaPlayer.play();});mediaPlayer.setAutoPlay(true);// 创建媒体视图MediaView mediaView = new MediaView(mediaPlayer);// 创建控制按钮Button playButton = new Button("播放");Button pauseButton = new Button("暂停");Button stopButton = new Button("停止");// 播放按钮点击事件playButton.setOnAction(event -> mediaPlayer.play());// 暂停按钮点击事件pauseButton.setOnAction(event -> mediaPlayer.pause());// 停止按钮点击事件stopButton.setOnAction(event -> mediaPlayer.stop());// 创建音量调节滑块Slider volumeSlider = new Slider(0, 1, 0.5);volumeSlider.setPrefWidth(100);mediaPlayer.volumeProperty().bind(volumeSlider.valueProperty());// 创建按钮布局HBox buttonBox = new HBox(10);buttonBox.setAlignment(Pos.CENTER);buttonBox.getChildren().addAll(playButton, pauseButton, stopButton);// 创建根布局BorderPane root = new BorderPane();root.setCenter(mediaView);root.setBottom(volumeSlider);root.setTop(buttonBox);// 创建场景Scene scene = new Scene(root, 800, 600);// 设置舞台primaryStage.setScene(scene);primaryStage.setTitle("Video Player");primaryStage.show();}
}
实体VideoInfoVO类:
package com.example.mybatisdemo02.model;import com.fasterxml.jackson.databind.ObjectMapper;
import lombok.Getter;
import lombok.Setter;@Getter
@Setter
public class VideoInfoVO {/*** 总帧数**/private int lengthInFrames;/*** 帧率**/private double frameRate;/*** 时长**/private double duration;/*** 视频编码*/private String videoCode;/*** 音频编码*/private String audioCode;private int width;private int height;private int audioChannel;private String md5;/*** 音频采样率*/private Integer sampleRate;private Double fileSize;public String toJson() {try {ObjectMapper objectMapper = new ObjectMapper();return objectMapper.writeValueAsString(this);} catch (Exception e) {return "";}}
}

调用:

public class Main {public static void main(String[] args) {VideoInfoVO videoInfoVO = FFmpegUtil.getVideoInfo("C:\\Users\\DELL\\Desktop\\test.mp4");int date = videoInfoVO.getLengthInFrames();double shichang = videoInfoVO.getDuration();System.out.println("帧数:"+date+"时长:"+shichang);}
}

结果如下图:

欢迎给个关注:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/791061.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

linux清理缓存垃圾命令和方法介绍

在Linux系统中&#xff0c;清理缓存和垃圾文件可以通过多种方法完成&#xff0c;这些方法旨在释放磁盘空间、提高系统性能。以下是一些常用的方法&#xff0c;结合了搜索结果中的信息&#xff1a; 1. 使用sync和echo命令清除RAM缓存和交换空间1 清除页面缓存&#xff08;Page …

【css】使用display:inline-block后,元素间存在4px的间隔

问题&#xff1a;在本地项目中使用【display: inline-block】&#xff0c;元素间存在4px间隔。打包后发布到外网又不存在这个问题了。 归根结底这是一个西文排版的问题&#xff0c;英文有空格作为词分界&#xff0c;而中文则没有。 此时的元素具有文本属性&#xff0c;只要标签…

基于深度学习的肿瘤图像检测系统(网页版+YOLOv8/v7/v6/v5代码+训练数据集)

摘要&#xff1a;在本博客中&#xff0c;我们深入探讨了基于YOLOv8/v7/v6/v5的肿瘤图像检测系统。核心上&#xff0c;我们采用了最新的YOLOv8技术&#xff0c;并将其与YOLOv7、YOLOv6、YOLOv5算法进行了综合整合和性能指标对比分析。我们详细阐述了当前国内外在此领域的研究现状…

Python机器学习实验 Python 数据分析

1.实验目的 掌握常见数据预处理方法&#xff0c;熟练运用数据分析方法&#xff0c;并掌握 Python 中的 Numpy、 Pandas 模块提供的数据分析方法。 2.实验内容 1. Pandas 基本数据处理 使用 Pandas 模块&#xff0c;完成以下操作。 &#xff08;1&#xff09;创建一个由 0 到 50…

Ai音乐大师演示(支持H5、小程序)独立部署源码

Ai音乐大师演示&#xff08;支持H5、小程序&#xff09;独立部署源码

政安晨:【Keras机器学习实践要点】(十五)—— KerasTuner 简述

目录 导言 调整模型结构 定义搜索空间 开始搜索 查询结果 重新训练模型 调整模型训练 调整数据预处理 重新训练模型 指定调整目标 以内置指标为目标 以自定义指标为目标 调整端到端工作流程 将 Keras 代码分开 政安晨的个人主页&#xff1a;政安晨 欢迎 &#x1…

React 入门

一、官网地址 英文官网: https://reactjs.org/中文官网: https://react.docschina.org/ 二、React 特点 声明式编码组件化编码React Native 编写原生应用高效&#xff08;优秀的 Diffing 算法&#xff09;高效的原因&#xff1a;1.使用虚拟DOM&#xff0c;不总是直接操作页面…

vultr ubuntu 服务器远程桌面安装及连接

一. 概述 vultr 上开启一个linux服务器&#xff0c;都是以终端形式给出的&#xff0c;默认不带 ui 桌面的&#xff0c;那其实对于想使用服务器上浏览器时的情形不是很好。那有没有方法在远程服务器安装桌面&#xff0c;然后原程使用呢&#xff1f;至少ubuntu的服务器是有的&am…

搜索--找出克隆二叉树中的相同节点

题目描述 给你两棵二叉树&#xff0c;原始树 original 和克隆树 cloned&#xff0c;以及一个位于原始树 original 中的目标节点 target。 其中&#xff0c;克隆树 cloned 是原始树 original 的一个 副本 。 请找出在树 cloned 中&#xff0c;与 target 相同 的节点&#xff…

Successive Convex Approximation算法的学习笔记

文章目录 一、代码debug二、原理 本文主要参考了CSDN上的 另一篇文章&#xff0c;但规范了公式的推导过程和修缮了部分代码 一、代码debug 首先&#xff0c;我们将所有的代码放到MATLAB中&#xff0c;很快在命令行中出现了错误信息 很显然有问题&#xff0c;但是我不知道发生…

redis 性能管理

一、查看 redis 内存使用 info memory 1&#xff0c; 进入 redis 查看 2&#xff0c; redis 外查看 二 内存碎片率 1&#xff0c;used_memory_rss 表示该进程所占物理内存的大小&#xff0c;即为操作系统分配给 Redis 实例的内存大小。 2&#xff0c;used_memory Redis …

【智能算法】猎豹优化器(CO)原理及实现

目录 1.背景2.算法原理2.1算法思想2.2算法过程 3.结果展示4.参考文献 1.背景 2022年&#xff0c;MA Akbari等人受到自然界中猎豹捕猎行为启发&#xff0c;提出了猎豹优化器&#xff08;The Cheetah Optimizer&#xff0c;CO&#xff09;。 2.算法原理 2.1算法思想 CO法对猎…

机器学习的模型校准

背景知识 之前一直没了解过模型校准是什么东西&#xff0c;最近上班业务需要看了一下&#xff1a; 模型校准是指对分类模型进行修正以提高其概率预测的准确性。在分类模型中&#xff0c;预测结果通常以类别标签形式呈现&#xff08;例如&#xff0c;0或1&#xff09;&#xf…

【THM】Nmap Advanced Port Scans(高级端口扫描)-初级渗透测试

介绍 本房间是Nmap系列的第三个房间(网络安全简介模块的一部分)。在前两个房间中,我们了解了实时主机发现和基本端口扫描。 Nmap实时主机发现Nmap基本端口扫描Nmap高级端口扫描Nmap后端口扫描在Nmap基本端口扫描中,我们介绍了TCP标志并回顾了TCP 3 路握手。要启动连接,TC…

AcWing刷题-约数个数

约数的个数 代码 # 计数 def f(x)->int:cnt 0i 1while i * i < x:if x % i 0:cnt 1if i * i < x:cnt 1i 1return cntn int(input()) a list(map(int,input().split())) for i in a:print(f(i))

蓝桥杯练习——拼出一个未来

选中 index.html 右键启动 Web Server 服务&#xff08;Open with Live Server&#xff09;&#xff0c;让项目运行起来。接着&#xff0c;打开环境右侧的【Web 服务】&#xff0c;就可以在浏览器中看到如下效果&#xff1a; 目标 完善 js/index.js 的 TODO 部分&#xff0c;实…

【leetcode】 c++ 数字全排列, test ok

1. 问题 2. 思路 3. 代码实现 #if 0 class Solution { private:vector<int> path; // 满足条件的一个结果 vector<vector<int>> res; // 结果集 void backtracking(vector<int> nums, vector<bool> used){// 若path的个数和nums个数相等&…

算法整理:排序

快速排序 首先不妨以第一个数为基准数&#xff0c;在一轮遍历后&#xff0c;使基准数左边的数都小于基准数&#xff0c;基准数右边的数都大于基准数。 当然也可以取中间的数为基准数。 void quick_sort(vector<int>&nums,int l,int r){if(l>r)return;int idxl;//…

硬件工程师职责与核心技能有哪些?

作为一个优秀的硬件工程师&#xff0c;必须要具备优秀的职业技能。那么&#xff0c;有些刚入行的工程师及在校的学生经常会问到&#xff1a;硬件工程师需要哪些核心技能&#xff1f;要回答这个问题&#xff0c;首先要明白硬件工程师的职责&#xff0c;然后才能知道核心技能要求…

神经网络学习笔记10——RNN、ELMo、Transformer、GPT、BERT

系列文章目录 参考博客1 参考博客2 文章目录 系列文章目录前言一、RNN1、简介2、模型结构3、RNN公式分析4、RNN的优缺点及优化1&#xff09;LSTM是RNN的优化结构2&#xff09;GRU是LSTM的简化结构 二、ELMo1、简介2、模型结构1&#xff09;输入2&#xff09;左右双向上下文信…