springboot(JavaCV )实现视频截取第N帧并保存图片

news/2024/12/22 14:48:36/

springboot(JavaCV )实现视频截取第N帧并保存图片

现在视频网站展示列表都是用img标签展示的,动图用的是gif,但是我们上传视频时并没有视屏封面,就这需要上传到服务器时自动生成封面并保存

本博客使用jar包的方式实现上传视频文件并且截取视频第一帧,保存到阿里云的OSS(也可以保存到本地获取其他任何地方)。

JavaCV 是一款开源的视觉处理库,基于GPLv2协议,对各种常用计算机视觉库封装后的一组jar包,

封装了OpenCV、libdc1394、OpenKinect、videoInput和ARToolKitPlus等计算机视觉编程人员常用库的接口。

此方法的好处是不需要再服务器上安装插件,直接代码中就可以实现视频截取。

我们需要截取视频第一帧,主要用到了ffmpeg和opencv。

一 , 引入jar包
我用到的maven的目前最新javacv版本,1.4.3,它应该支持jdk1.7及以上,我项目用的还是jdk1.8.

不过需要注意的是在使用的过程当中 , maven引入jar的时候 会引入所有平台的版本

全部引入大小在五百兆左右(不建议使用)

<!--视频截取第一帧--><dependency><groupId>org.bytedeco</groupId><artifactId>javacv</artifactId><version>1.4.3</version></dependency><dependency><groupId>org.bytedeco.javacpp-presets</groupId><artifactId>ffmpeg-platform</artifactId><version>4.0.2-1.4.3</version></dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-test</artifactId><version>RELEASE</version></dependency>

二 , java 代码实现

public class ImgTools {//util调用application.propertiesprivate final static ResourceBundle RESOURCE_BUNDLE = ResourceBundle.getBundle("application");private final static String aliyuVideonImg = RESOURCE_BUNDLE.getString("aliyun.video.img");//    public static void main(String[] args) throws Exception {
//        ImgTools imgTools = new ImgTools();
//        System.out.println(imgTools.randomGrabberFFmpegVideoImage
//                ("视频地址,可以是网络视频,也可以是本地视频"));
//    }/*** 获取视频缩略图** @param filePath:视频路径* @throws Exception*/public String randomGrabberFFmpegVideoImage(String filePath) throws Exception {String targetFilePath = "";FFmpegFrameGrabber ff = FFmpegFrameGrabber.createDefault(filePath);ff.start();//判断是否是竖屏小视频String rotate = ff.getVideoMetadata("rotate");int ffLength = ff.getLengthInFrames();Frame f;int i = 0;int index = 3;//截取图片第几帧while (i < ffLength) {f = ff.grabImage();if (i == index) {if (null != rotate && rotate.length() > 1) {targetFilePath = doExecuteFrame(f, true);   //获取缩略图} else {targetFilePath = doExecuteFrame(f, false);   //获取缩略图}break;}i++;}ff.stop();return targetFilePath;  //返回的是视频第N帧}/*** 截取缩略图,存入阿里云OSS(按自己的上传类型自定义转换文件格式)** @param f* @return* @throws Exception*/public String doExecuteFrame(Frame f, boolean bool) throws Exception {if (null == f || null == f.image) {return "";}Java2DFrameConverter converter = new Java2DFrameConverter();BufferedImage bi = converter.getBufferedImage(f);if (bool == true) {Image image = (Image) bi;bi = rotate(image, 90);//图片旋转90度}ByteArrayOutputStream os = new ByteArrayOutputStream();ImageIO.write(bi, "png", os);byte[] sdf = os.toByteArray();InputStream input = new ByteArrayInputStream(os.toByteArray());MultipartFile multipartFile = new MockMultipartFile("temp.jpg", "temp.jpg", "temp.jpg", input);Aliyunoss aliyunoss = new Aliyunoss();//如需了解阿里云OSS,请详读我的另一篇博客("https://blog.csdn.net/weixin_44401989/article/details/105732856")String url = aliyunoss.uploadAli(multipartFile, aliyuVideonImg);return url;}/*** 图片旋转角度** @param src   源图片* @param angel 角度* @return 目标图片*/public static BufferedImage rotate(Image src, int angel) {int src_width = src.getWidth(null);int src_height = src.getHeight(null);// calculate the new image sizeRectangle rect_des = CalcRotatedSize(new Rectangle(new Dimension(src_width, src_height)), angel);BufferedImage res = null;res = new BufferedImage(rect_des.width, rect_des.height,BufferedImage.TYPE_INT_RGB);Graphics2D g2 = res.createGraphics();// transform(这里先平移、再旋转比较方便处理;绘图时会采用这些变化,绘图默认从画布的左上顶点开始绘画,源图片的左上顶点与画布左上顶点对齐,然后开始绘画,修改坐标原点后,绘画对应的画布起始点改变,起到平移的效果;然后旋转图片即可)//平移(原理修改坐标系原点,绘图起点变了,起到了平移的效果,如果作用于旋转,则为旋转中心点)g2.translate((rect_des.width - src_width) / 2, (rect_des.height - src_height) / 2);//旋转(原理transalte(dx,dy)->rotate(radians)->transalte(-dx,-dy);修改坐标系原点后,旋转90度,然后再还原坐标系原点为(0,0),但是整个坐标系已经旋转了相应的度数 )g2.rotate(Math.toRadians(angel), src_width / 2, src_height / 2);//        //先旋转(以目标区域中心点为旋转中心点,源图片左上顶点对准目标区域中心点,然后旋转)
//        g2.translate(rect_des.width/2,rect_des.height/ 2);
//        g2.rotate(Math.toRadians(angel));
//        //再平移(原点恢复到源图的左上顶点处(现在的右上顶点处),否则只能画出1/4)
//        g2.translate(-src_width/2,-src_height/2);g2.drawImage(src, null, null);return res;}/*** 计算转换后目标矩形的宽高** @param src   源矩形* @param angel 角度* @return 目标矩形*/private static Rectangle CalcRotatedSize(Rectangle src, int angel) {double cos = Math.abs(Math.cos(Math.toRadians(angel)));double sin = Math.abs(Math.sin(Math.toRadians(angel)));int des_width = (int) (src.width * cos) + (int) (src.height * sin);int des_height = (int) (src.height * cos) + (int) (src.width * sin);return new java.awt.Rectangle(new Dimension(des_width, des_height));}
}
public class ImgTools {//util调用application.propertiesprivate final static ResourceBundle RESOURCE_BUNDLE = ResourceBundle.getBundle("application");private final static String aliyuVideonImg = RESOURCE_BUNDLE.getString("aliyun.video.img");//    public static void main(String[] args) throws Exception {
//        ImgTools imgTools = new ImgTools();
//        System.out.println(imgTools.randomGrabberFFmpegVideoImage
//                ("视频地址,可以是网络视频,也可以是本地视频"));
//    }/*** 获取视频缩略图** @param filePath:视频路径* @throws Exception*/public String randomGrabberFFmpegVideoImage(String filePath) throws Exception {String targetFilePath = "";FFmpegFrameGrabber ff = FFmpegFrameGrabber.createDefault(filePath);ff.start();//判断是否是竖屏小视频String rotate = ff.getVideoMetadata("rotate");int ffLength = ff.getLengthInFrames();Frame f;int i = 0;int index = 3;//截取图片第几帧while (i < ffLength) {f = ff.grabImage();if (i == index) {if (null != rotate && rotate.length() > 1) {targetFilePath = doExecuteFrame(f, true);   //获取缩略图} else {targetFilePath = doExecuteFrame(f, false);   //获取缩略图}break;}i++;}ff.stop();return targetFilePath;  //返回的是视频第N帧}/*** 截取缩略图,存入阿里云OSS(按自己的上传类型自定义转换文件格式)** @param f* @return* @throws Exception*/public String doExecuteFrame(Frame f, boolean bool) throws Exception {if (null == f || null == f.image) {return "";}Java2DFrameConverter converter = new Java2DFrameConverter();BufferedImage bi = converter.getBufferedImage(f);if (bool == true) {Image image = (Image) bi;bi = rotate(image, 90);//图片旋转90度}ByteArrayOutputStream os = new ByteArrayOutputStream();ImageIO.write(bi, "png", os);byte[] sdf = os.toByteArray();InputStream input = new ByteArrayInputStream(os.toByteArray());MultipartFile multipartFile = new MockMultipartFile("temp.jpg", "temp.jpg", "temp.jpg", input);Aliyunoss aliyunoss = new Aliyunoss();//如需了解阿里云OSS,请详读我的另一篇博客("https://blog.csdn.net/weixin_44401989/article/details/105732856")String url = aliyunoss.uploadAli(multipartFile, aliyuVideonImg);return url;}/*** 图片旋转角度** @param src   源图片* @param angel 角度* @return 目标图片*/public static BufferedImage rotate(Image src, int angel) {int src_width = src.getWidth(null);int src_height = src.getHeight(null);// calculate the new image sizeRectangle rect_des = CalcRotatedSize(new Rectangle(new Dimension(src_width, src_height)), angel);BufferedImage res = null;res = new BufferedImage(rect_des.width, rect_des.height,BufferedImage.TYPE_INT_RGB);Graphics2D g2 = res.createGraphics();// transform(这里先平移、再旋转比较方便处理;绘图时会采用这些变化,绘图默认从画布的左上顶点开始绘画,源图片的左上顶点与画布左上顶点对齐,然后开始绘画,修改坐标原点后,绘画对应的画布起始点改变,起到平移的效果;然后旋转图片即可)//平移(原理修改坐标系原点,绘图起点变了,起到了平移的效果,如果作用于旋转,则为旋转中心点)g2.translate((rect_des.width - src_width) / 2, (rect_des.height - src_height) / 2);//旋转(原理transalte(dx,dy)->rotate(radians)->transalte(-dx,-dy);修改坐标系原点后,旋转90度,然后再还原坐标系原点为(0,0),但是整个坐标系已经旋转了相应的度数 )g2.rotate(Math.toRadians(angel), src_width / 2, src_height / 2);//        //先旋转(以目标区域中心点为旋转中心点,源图片左上顶点对准目标区域中心点,然后旋转)
//        g2.translate(rect_des.width/2,rect_des.height/ 2);
//        g2.rotate(Math.toRadians(angel));
//        //再平移(原点恢复到源图的左上顶点处(现在的右上顶点处),否则只能画出1/4)
//        g2.translate(-src_width/2,-src_height/2);g2.drawImage(src, null, null);return res;}/*** 计算转换后目标矩形的宽高** @param src   源矩形* @param angel 角度* @return 目标矩形*/private static Rectangle CalcRotatedSize(Rectangle src, int angel) {double cos = Math.abs(Math.cos(Math.toRadians(angel)));double sin = Math.abs(Math.sin(Math.toRadians(angel)));int des_width = (int) (src.width * cos) + (int) (src.height * sin);int des_height = (int) (src.height * cos) + (int) (src.width * sin);return new java.awt.Rectangle(new Dimension(des_width, des_height));}
}

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

相关文章

从电子表格到纸张:Excel转PDF的神奇变身之旅!

当你需要将Excel文件转换为PDF时&#xff0c;可以使用Python编程语言和一些流行的库来实现这个任务。在本篇博客中&#xff0c;我将介绍如何使用wxPython、pandas和PyMuPDF库创建一个简单易用的图形用户界面(GUI)工具来完成这项工作。 C:\pythoncode\new\excelexportpdf.py …

【云原生】Docker Cgroups资源控制管理

目录 一、cgroups简介 cgroups有四大功能&#xff1a; 二、cpu时间片的概念 三、对CPU使用的限制 3.1 设置CPU使用率上限 &#xff08;1&#xff09;查看容器的默认CPU使用限制 &#xff08;2&#xff09;进行压力测试 &#xff08;3&#xff09;创建容器时设置CPU使用时…

【从零学习python 】56. 异常处理在程序设计中的重要性与应用

文章目录 异常的概念读取文件异常try...except语句try...else语句try...finally语句 进阶案例 异常的概念 在程序运行过程中&#xff0c;由于编码不规范或其他客观原因&#xff0c;可能会导致程序无法继续运行&#xff0c;此时就会出现异常。如果不对异常进行处理&#xff0c;…

Linux Kernel 4.12 或将新增优化分析工具

到 7 月初&#xff0c;Linux Kernel 4.12 预计将为修复所有安全漏洞而奠定基础&#xff0c;另外新增的是一个分析工具&#xff0c;对于开发者优化启动时间时会有所帮助。 新的「个别任务统一模型」&#xff08;Per-Task Consistency Model&#xff09;为主要核心实时修补&#…

XXX程序 详细说明

用于记录理解PC程序的程序逻辑 1、程序的作用 根据原作者的说明&#xff08;文件说明.txt&#xff09;&#xff0c;该程序 (PC.py) 的主要作用是提取某一个文件夹中的某个设备 (通过config中的信息看出来是Ag_T_8) 产生的日志文件&#xff0c;然后提取其中某些需要的数据&…

开源在线图片设计器,支持PSD解析、AI抠图等,基于Puppeteer生成图片

Github 开源地址: palxiao/poster-design 项目速览 git clone https://github.com/palxiao/poster-design.git cd poster-design npm run prepared # 快捷安装依赖指令 npm run serve # 本地运行将同时运行前端界面与图片生成服务(3000与7001端口)&#xff0c;合成图片时…

SOFARPC(笔记)

文章目录 一、快速开始1.1 SOFARPC1.2 基于SOFABoot 二、注册中心三、通讯协议2.1 Bolt基本发布调用方式超时控制协议泛化调用序列化协议自定义线程池 2.2 RESTful基本使用 2.3 其他协议四、架构 附录 官方样例下载地址-sofa-boot-guides 可查看 SOFARPC 方式快速入门 一、快…

快速解决Spring Boot跨域困扰:使用CORS实现无缝跨域支持

跨域问题 什么是跨域&#xff1f; 跨域&#xff08;Cross-Origin Issue&#xff09;的存在是因为浏览器的安全限制&#xff0c;它防止恶意网站利用跨域请求来获取用户的敏感信息或执行恶意操作。浏览器通过实施同源策略来限制网页在不同源之间进行资源访问或交互的情况。当一…