StringBoot 通过FFmpegFrameGrabber获取视频首帧作为图片展示

Posted 揽月随风醉

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了StringBoot 通过FFmpegFrameGrabber获取视频首帧作为图片展示相关的知识,希望对你有一定的参考价值。

FFmpegFrameGrabber:点击了解更多使用方法

一、导包

<!--获取视频首帧作为图片展示-->
<dependency>
    <groupId>org.bytedeco</groupId>
    <artifactId>javacpp</artifactId>
    <version>1.4.1</version>
</dependency>
<dependency>
    <groupId>org.bytedeco</groupId>
    <artifactId>javacv</artifactId>
    <version>1.4.1</version>
</dependency>
<dependency>
    <groupId>org.bytedeco.javacpp-presets</groupId>
    <artifactId>ffmpeg-platform</artifactId>
    <version>3.4.2-1.4.1</version>
</dependency>

二、代码

package com.xmtdxt.pc.api.config;

import org.bytedeco.javacv.FFmpegFrameGrabber;
import org.bytedeco.javacv.Frame;
import org.bytedeco.javacv.Java2DFrameConverter;

import javax.imageio.ImageIO;
import java.awt.image.BufferedImage;
import java.awt.image.RenderedImage;
import java.io.File;
import java.util.ArrayList;
import java.util.List;

/**
 * 获取视频首帧图片用于界面展示
 *
 * @author LunarYouI
 * 基本信息描述:https://www.cnblogs.com/liangjingfu/p/12858018.html
 *
 * Input #0, mov,mp4,m4a,3gp,3g2,mj2, from 'F:\\temporary\\123456789.mp4':
 *   Metadata:
 *     major_brand     : isom
 *     minor_version   : 512
 *     compatible_brands: isomiso2avc1mp41
 *     encoder         : Lavf58.12.100
 *     description     : Tencent SID MTS
 *   Duration: 00:04:13.21, start: 0.000000, bitrate: 3167 kb/s
 *     Stream #0:0(und): Video: h264 (High) (avc1 / 0x31637661), yuv420p, 1280x720 [SAR 1:1 DAR 16:9], 3033 kb/s, 24 fps, 24 tbr, 90k tbn, 48 tbc (default)
 *     Metadata:
 *       handler_name    : VideoHandler
 *     Stream #0:1(und): Audio: aac (LC) (mp4a / 0x6134706D), 44100 Hz, stereo, fltp, 128 kb/s (default)
 *     Metadata:
 *       handler_name    : SoundHandler
 *
 *
 * @create 2022-06-13 9:41
 */
public class GetVideoGainImg 
    /**
     *
     * @Title: getTempPath
     * @Description: 生成视频的首帧图片方法
     * @author: Zing
     * @param: @param tempPath 生成首帧图片的文件地址
     * @param: @param filePath 传进来的线上文件
     * @param: @return
     * @param: @throws Exception
     * @return: boolean
     * @throws
     */
    public static boolean getTempPath(String tempPath, String filePath) throws Exception 
        File targetFile = new File(tempPath);
        if (!targetFile.getParentFile().exists()) 
            targetFile.getParentFile().mkdirs();
        
        File file2 = new File(filePath);
        System.out.println("确认文件是否是视频...");
        //判断文件是否为视频
        if(GetVideoGainImg.isVideo(filePath)) 
            System.out.println("确认成功!");
            //判断文件是否存在
            if (file2.getParentFile().exists()) 
                FFmpegFrameGrabber ff = new FFmpegFrameGrabber(file2);
                ff.start();
                int ftp = ff.getLengthInFrames();
                int flag=0;
                Frame frame = null;
                while (flag <= ftp) 
                    //获取帧
                    frame = ff.grabImage();
                    //过滤前3帧,避免出现全黑图片
                    if ((flag>3)&&(frame != null)) 
                        break;
                    
                    flag++;
                
                if(ImageIO.write(FrameToBufferedImage(frame), "jpg", targetFile)) 
	                ff.close();
                    ff.stop();
                    System.out.println("输出图片成功!");
                    return true;
                else 
	                ff.close();
                    ff.stop();
                    System.out.println("输出图片失败!");
                    return false;
                
            else
                System.out.println("路径内容错误!"+file2.getParentFile());
                return false;
            
        else
            System.out.println("确认失败!");
            return false;
        
    

    /***
     *
     * @Title: FrameToBufferedImage
     * @Description: 创建格式化BufferedImage对象
     * @author: Zing
     * @param: @param frame
     * @param: @return
     * @return: RenderedImage
     * @throws
     */
    private static RenderedImage FrameToBufferedImage(Frame frame) 
        //创建BufferedImage对象
        Java2DFrameConverter converter = new Java2DFrameConverter();
        BufferedImage bufferedImage = converter.getBufferedImage(frame);
        return bufferedImage;
    

    /**
     *
     * @Title: isVideo
     * @Description:判断是不是视频
     * @author: Zing
     * @param: @param path 文件路径
     * @param: @return
     * @return: boolean       true是视频 false非视频
     * @throws
     */
    public static boolean isVideo(String path) 
        //设置视频后缀
        List<String> typeList = new ArrayList<String>();
        typeList.add("mp4");
        typeList.add("flv");
        typeList.add("avi");
        typeList.add("rmvb");
        typeList.add("rm");
        typeList.add("wmv");
        //获取文件名和后缀
        String suffix = path.substring(path.lastIndexOf(".") + 1);
        for(String type : typeList) 
            //统一为大写作比较
            if(type.toUpperCase().equals(suffix.toUpperCase())) 
                return true;
            
        
        return false;
    

    public static void main(String[] args) throws Exception 
        boolean tempPath = GetVideoGainImg.getTempPath("F:/temporary/542.jpg", "F:/temporary/123456789.mp4");
        System.out.println("获取首帧是否成功!"+tempPath);
    


三、基本信息描述

更多信息查看—》点击跳转

这行信息表明了该文件的Metadata信息
比如encoder是 Lavf58.12.100 ,
Lavf 代表的是FFmpeg输出的文件
58.12.100 代表了FFmpeg的版本代号

Metadata:
    major_brand     : isom
    minor_version   : 512
    compatible_brands: isomiso2avc1mp41
    encoder         : Lavf58.12.100
    description     : Tencent SID MTS

该文件的时长是 00:04:13.21【时:分:秒.毫秒】
开始播放时间是 0
整个文件的比特率是 3167 kb/s
(视频比特率是指每秒传送的比特(bit)数。单位为bps(Bit Per Second),比特率越高,每秒传送数据就越多,画质就越清晰)

第一个 stream 是视频流
编码方式是 h264 的格式(封装格式是avc1)
每一帧的数据表示是 YUV420P 的格式
分辨率是 1280x720
这路流的比特率是 3033 kb/s
帧率是每秒钟 24帧(24 fps)
fps 表示平均帧率,总帧数除以总时长(以s为单位)。
tbr 表示帧率,该参数倾向于一个基准,往往tbr跟fps相同。
tbn 表示视频流 timebase(时间基准),
比如ts流的timebase 为90000,flv格式视频流timebase为1000,tbc 表示视频流codec timebase ,对于264码流该参数通过解析sps间接获取(通过sps获取帧率)。


第二个 stream 是音频流
编码方式是 aac(封装格式是mp4a)
采用的 Profile 是 LC规格
采用率是 44100 Hz
声道数是立体声
数据表示格式是浮点型
这路音频流的比特率是 128 kb/s

Stringboot项目打war包

正常的boot项目打包打的是jar包,boot项目内置tomcat,一般我打包是:mvn clean install -Dmaven.test.skip 命令

clean 移除所有上一次构建生成的文件

install 将包安装至本地仓库,以让其它项目依赖

maven.test.skip 跳过测试

运行的命令:java -jar xxx.jar

       由于一些外部的原因我们需要将项目打包成war包,这时候我们可以创建一个新的boot项目选择war包的方式进行参考,这种方式的是spring官方推荐的打包方式,其实打war包也就进行了一下几步操作:

1、在pom.xml文件首部增加<packaging>war</packaging>,/
<groupId>com.uloan.ssm</groupId>
<artifactId>Uloan</artifactId>
<version>1.0-SNAPSHOT</version>
<packaging>war</packaging>
<!--<packaging>jar</packaging>-->
2、<!--增加下面的依赖覆盖默认内嵌的Tomcat依赖-->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-tomcat</artifactId>
    <scope>provided</scope>
</dependency>
scope的分类
1.compile:默认值 他表示被依赖项目需要参与当前项目的编译,还有后续的测试,运行周期也参与其中,是一个比较强的依赖。打包的时候通常需要包含进去
2.test:依赖项目仅仅参与测试相关的工作,包括测试代码的编译和执行,不会被打包,例如:junit
3.runtime:表示被依赖项目无需参与项目的编译,不过后期的测试和运行周期需要其参与。与compile相比,跳过了编译而已。例如JDBC驱动,适用运行和测试阶段
4.provided:打包的时候可以不用包进去,别的设施会提供。事实上该依赖理论上可以参与编译,测试,运行等周期。相当于compile,但是打包阶段做了exclude操作
5.system:从参与度来说,和provided相同,不过被依赖项不会从maven仓库下载,而是从本地文件系统拿。需要添加systemPath的属性来定义路径

所以我们这里采用provided,这也是spring官方推荐的方式。

剩下的就是将我们的启动类进行修改,我们之前启动类是boot的启动方式,我们这边要修改成tomcat的方式,当然之前的boot启动方式在我们本地依旧保留使用。

public class ServletInitializer extends SpringBootServletInitializer {

    @Override
    protected SpringApplicationBuilder configure(SpringApplicationBuilder application) {
        return application.sources(UloanApplication.class);
    }
}

该类继承了SpringBootServletInitializer并且重写了configure方法。

jar包和war包启动区别
    jar包:执行SpringBootApplication的run方法,启动IOC容器,然后创建嵌入式Servlet容器
    war包:  先是启动Servlet服务器,服务器启动Springboot应用(springBootServletInitizer),然后启动IOC容器
SpringBootServletInitializer实例执行onStartup方法的时候会通过createRootApplicationContext方法来执行run方法,接下来的过程就同以jar包形式启动的应用的run过程一样了,在内部会创建IOC容器并返回,只是以war包形式的应用在创建IOC容器过程中,不再创建Servlet容器了。

























以上是关于StringBoot 通过FFmpegFrameGrabber获取视频首帧作为图片展示的主要内容,如果未能解决你的问题,请参考以下文章

StringBoot 易上手的java框架

StringBoot入门

StringBoot集成Elasticsearch

StringBoot整合Mytais登录案例

Stringboot 拦截器Filter实现登录认证

Stringboot项目打war包