java获取近期视频流关键帧与截图

1、背景

最近在做视频转发的开发时,遇到一个问题,前端订阅播放h264视频流时,有时会出现一段时间黑屏,经过测试发现是没有收到关键帧,只有第一帧是关键帧才能保证后续播放正常。所以后端需要实现一个功能,就是前端在进入播放页面时,后端把最近的一个关键帧发过去。

2、思路(环形缓存区)

后端接收到的视频流是一个个的字节数组,所以在接收时没法直接判断一帧的开始和结束,需要将最近的一段视频流截取出来,然后利用ffmpeg工具进行整体的解析和关键帧提取。

查看ffmpeg工具的代码,可以发现ffmpeg工具入参是inputstream,该工具会不断调用inputstream的read方法进行字节的读取。所以就想通过一个环形缓存区不断的记录最新的一段视频流数据。该环形缓存区再实现inputstream的接口,重写read方法,read读取的开始位置即是环形缓存区头,到环形缓存区的尾时自动结束。

3、依赖包

XML 复制代码
<!--   javacv    -->
<dependency>
	<groupId>org.bytedeco</groupId>
	<artifactId>javacpp</artifactId>
	<version>1.4.3</version>
</dependency>
<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>

4、环形缓存区定义

java 复制代码
public class CycleBufferInputStream extends InputStream {

    /************************************************  环形缓冲区  ***********************************************/
    private ByteBuffer buffer = null;

    private int readPos = 0; //将要读取的位置
    private int writePos = 0; //将要写入的位置
    private boolean isCycle = false; //判断是否已经形成一个环

    public CycleBufferInputStream(int capacity) {
        this.buffer = ByteBuffer.allocateDirect(capacity);
    }

    /**
     * 将字节数组以覆盖的方式放入环形缓冲区
     */
    public void put(byte[] bytes) {
        int used = buffer.capacity() - buffer.position();
        if (used < bytes.length) {
            buffer.put(bytes, 0, used);
            buffer.clear();
            buffer.put(bytes, used, bytes.length - used);
            isCycle = true;
        } else if (used == bytes.length) {
            buffer.put(bytes, 0, used);
            buffer.clear();
            isCycle = true;
        } else {
            buffer.put(bytes, 0, bytes.length);
        }
        writePos = buffer.position();
    }

    /**
     * 定位读取的初始位置(执行inputstream 读取前,必须要先调用该方法)
     */
    public void readPrepare() {
        if (buffer.capacity() == writePos || !isCycle) {
            readPos = 0;
        } else {
            readPos = buffer.position() + 1;
        }
    }

    /*************************************************** 输入流传输  ***************************************************/

    /**
     * Reads the next byte of data from the input stream. The value byte is returned as an int in the range 0 to 255. If no byte is available because the end of the stream has been reached, the value -1 is returned. This method blocks until input data is available, the end of the stream is detected, or an exception is thrown.
     * A subclass must provide an implementation of this method.
     * Returns: the next byte of data, or -1 if the end of the stream is reached.
     * Throws: IOException -- if an I/O error occurs.
     */
    @Override
    public int read() throws IOException {
        if (readPos == buffer.capacity()) readPos = 0;
        if (readPos == writePos) return -1;
        int value = buffer.get(readPos++);
        if (value < 0) value = value + 256;
        return value;
    }

}

5、从环形缓存区提取关键帧

java 复制代码
    /**
     * 从环形缓存环获取最近一帧关键帧字节数组
     * 这里返回的堆外内存,所以注意要及时进行内存释放
     * @param inputStream
     */
    public static ByteBuffer dealVideo(InputStream inputStream)  {
        try {
            int j = 0;
            FFmpegFrameGrabber ff = new FFmpegFrameGrabber(inputStream);
            ff.start();
            Frame frame = null;
            Frame last = null;
            while ((frame = ff.grabKeyFrame()) != null && frame.image != null) {
                last = frame.clone();
                System.out.println("获取一帧" + j++);
            }
            ff.stop();

            if (last != null) {
                System.out.println("获取最近的一个关键帧");
                ByteBuffer byteBuffer = (ByteBuffer)last.image[0];
                return byteBuffer;
            }
        } catch (Exception e) {
            log.error("提取最近一个关键帧异常\n",e);
        }
        return null;
    }

6、使用

java 复制代码
    public static void main(String[] args) throws Exception {
        CycleBufferInputStream stream = new CycleBufferInputStream(1024 * 1024 * 10);

        FileInputStream fis = new FileInputStream("D:\\tmp-data\\1694511149969.h264");
        byte[] bytes = new byte[fis.available()];
        fis.read(bytes);
        stream.put(bytes);

        stream.readPrepare();
        dealVideo(stream);
    }

7、扩展:获取近期视频的截图

其实就是从近期的关键帧中提取出图片,关键代码如下:

java 复制代码
    public static void dealImage(InputStream inputStream) {
        try {
            Java2DFrameConverter converter = new Java2DFrameConverter();

            FFmpegFrameGrabber ff = new FFmpegFrameGrabber(inputStream);
            ff.start();
            int j = 0;
            Frame frame = null;
            Frame last = null;
            while ((frame = ff.grabImage()) != null) {
                last = frame.clone();
                System.out.println("获取一张图片" + j++);
            }
            ff.stop();

            if (last != null && last.image != null) {
                System.out.println("存储最后一张图片 ");
                BufferedImage fecthedImage = converter.getBufferedImage(last);
                File screenshotFile = new File("D:\\tmp-data\\", System.currentTimeMillis() + ".jpg");
                ImageIO.write(fecthedImage, "jpg", screenshotFile);
            }

        } catch (Exception e) {
            log.error("提取最近一个图片异常\n",e);
        }
    }

8、整体代码

java 复制代码
package com.qq.utils;

import lombok.extern.slf4j.Slf4j;
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.io.*;
import java.nio.ByteBuffer;

@Slf4j
public class CycleBufferInputStream extends InputStream {

    /************************************************  环形缓冲区  ***********************************************/
    private ByteBuffer buffer = null;

    private int readPos = 0; //将要读取的位置
    private int writePos = 0; //将要写入的位置
    private boolean isCycle = false; //判断是否已经形成一个环

    public CycleBufferInputStream(int capacity) {
        this.buffer = ByteBuffer.allocateDirect(capacity);
    }

    /**
     * 将字节数组以覆盖的方式放入环形缓冲区
     */
    public void put(byte[] bytes) {
        int used = buffer.capacity() - buffer.position();
        if (used < bytes.length) {
            buffer.put(bytes, 0, used);
            buffer.clear();
            buffer.put(bytes, used, bytes.length - used);
            isCycle = true;
        } else if (used == bytes.length) {
            buffer.put(bytes, 0, used);
            buffer.clear();
            isCycle = true;
        } else {
            buffer.put(bytes, 0, bytes.length);
        }
        writePos = buffer.position();
    }

    /**
     * 定位读取的初始位置(执行inputstream方法前,必须要先调用该方法)
     */
    public void readPrepare() {
        if (buffer.capacity() == writePos || !isCycle) {
            readPos = 0;
        } else {
            readPos = buffer.position() + 1;
        }
    }

    /*************************************************** 输入流传输  ***************************************************/
    @Override
    public int read() throws IOException {
        if (readPos == buffer.capacity()) readPos = 0;
        if (readPos == writePos) return -1;
        int value = buffer.get(readPos++);
        if (value < 0) value = value + 256;
        return value;
    }

    /**
     * 从环形缓存环获取最近一帧关键帧字节数组
     * 这里返回的堆外内存,所以注意要及时进行内存释放
     * @param inputStream
     */
    public static ByteBuffer dealVideo(InputStream inputStream)  {
        try {
            int j = 0;
            FFmpegFrameGrabber ff = new FFmpegFrameGrabber(inputStream);
            ff.start();
            Frame frame = null;
            Frame last = null;
            while ((frame = ff.grabKeyFrame()) != null && frame.image != null) {
                last = frame.clone();
                System.out.println("获取一帧" + j++);
            }
            ff.stop();

            if (last != null) {
                System.out.println("获取最近的一个关键帧");
                ByteBuffer byteBuffer = (ByteBuffer)last.image[0];
                return byteBuffer;
            }
        } catch (Exception e) {
            log.error("提取最近一个关键帧异常\n",e);
        }
        return null;
    }

    public static void dealImage(InputStream inputStream) {
        try {
            Java2DFrameConverter converter = new Java2DFrameConverter();

            FFmpegFrameGrabber ff = new FFmpegFrameGrabber(inputStream);
            ff.start();
            int j = 0;
            Frame frame = null;
            Frame last = null;
            while ((frame = ff.grabImage()) != null) {
                last = frame.clone();
                System.out.println("获取一张图片" + j++);
            }
            ff.stop();

            if (last != null && last.image != null) {
                System.out.println("存储最后一张图片 ");
                BufferedImage fecthedImage = converter.getBufferedImage(last);
                File screenshotFile = new File("D:\\tmp-data\\", System.currentTimeMillis() + ".jpg");
                ImageIO.write(fecthedImage, "jpg", screenshotFile);
            }

        } catch (Exception e) {
            log.error("提取最近一个图片异常\n",e);
        }
    }

}
相关推荐
转世成为计算机大神2 分钟前
易考八股文之Java中的设计模式?
java·开发语言·设计模式
qq_3273427323 分钟前
Java实现离线身份证号码OCR识别
java·开发语言
阿龟在奔跑2 小时前
引用类型的局部变量线程安全问题分析——以多线程对方法局部变量List类型对象实例的add、remove操作为例
java·jvm·安全·list
飞滕人生TYF2 小时前
m个数 生成n个数的所有组合 详解
java·递归
代码小鑫2 小时前
A043-基于Spring Boot的秒杀系统设计与实现
java·开发语言·数据库·spring boot·后端·spring·毕业设计
真心喜欢你吖2 小时前
SpringBoot与MongoDB深度整合及应用案例
java·spring boot·后端·mongodb·spring
激流丶2 小时前
【Kafka 实战】Kafka 如何保证消息的顺序性?
java·后端·kafka
周全全2 小时前
Spring Boot + Vue 基于 RSA 的用户身份认证加密机制实现
java·vue.js·spring boot·安全·php
uzong3 小时前
一个 IDEA 老鸟的 DEBUG 私货之多线程调试
java·后端
AiFlutter3 小时前
Java实现简单的搜索引擎
java·搜索引擎·mybatis