当前位置: 首页 > news >正文

ffmpeg视频解码

一、视频解码流程

使用ffmpeg解码视频帧主要可分为两大步骤:初始化解码器解码视频帧,以下代码以mjpeg为例

1. 初始化解码器

初始化解码器主要有以下步骤:

(1)查找解码器

// 查找MJPEG解码器pCodec = avcodec_find_decoder_by_name(videoCodecName);if (pCodec == nullptr) {release();return false;}// 分配解码器上下文pCodecCtx = avcodec_alloc_context3(pCodec);if (!pCodecCtx) {release();return false;}

(2)设置解码器参数

    pCodecCtx->codec_type = AVMEDIA_TYPE_VIDEO;pCodecCtx->width = mVideoSrcWidth; // 视频宽度pCodecCtx->height = mVideoSrcHeight; // 视频高度pCodecCtx->pix_fmt = AV_PIX_FMT_YUVJ422P; // 或者其他适合的格式pCodecCtx->time_base = { 1, mVideoSrcFps }; // 帧率pCodecCtx->thread_count = 2;

(3)打开解码器

 // 打开解码器if (avcodec_open2(pCodecCtx, pCodec, nullptr) < 0) {release();return false;}

2. 解码视频帧数据

解码视频帧数据主要有以下步骤:

(1)将编码数据送往解码器

AVPacket* packet = av_packet_alloc();if (!packet) {release();return false;}packet->data = data; // 待解码数据地址packet->size = size; // 待解码数据大小// 发送数据到解码器int ret = avcodec_send_packet(pCodecCtx, packet);if (ret < 0) {release();return false;}

(2)接收解码数据

ret = avcodec_receive_frame(pCodecCtx, pFrame);

二、使用ffmpeg实现对内存中的视频帧数据解码

以下代码中InitDecoder为初始化解码器接口,DecodeVideoFrame为解码视频帧接口

需注意

(1)解码的色彩空间pCodecCtx->pix_fmt不可随意指定,调用avcodec_send_packet后可能会变化,这与视频帧的编码方式有关
(2)每次接收完解码数据要调用av_frame_unref进行释放,否则会有内存泄漏问题


extern "C" { // ffmpeg为使用C语言库,因此要声明为C语言的方式链接
#include <libavformat/avformat.h>
#include <libavcodec/avcodec.h>
#include <libavutil/avutil.h>
#include <libavutil/opt.h>
#include <libavutil/mathematics.h>
#include <libavutil/timestamp.h>
#include <libswresample/swresample.h>
#include <libswscale/swscale.h>
#include <libavutil/imgutils.h>
#include <libavutil/error.h>
}char videoCodecName[] = "mjpeg";
FILE* out_file = nullptr;AVFormatContext* pFormatCtx = nullptr;AVCodecContext* pCodecCtx = nullptr;
const AVCodec* pCodec = nullptr;
AVFrame* pFrame = nullptr;
AVFrame* pFrameYUYV = nullptr;
int yuyv_size = 0;
uint8_t* buffer = nullptr;
SwsContext* sws_ctx = nullptr;int      mVideoSrcWidth = 1920;
int      mVideoSrcHeight = 1080;
int      mVideoSrcFps = 30;bool InitDecoder() {fopen_s(&out_file, "test_yuv.yuv", "wb");// 查找MJPEG解码器pCodec = avcodec_find_decoder_by_name(videoCodecName);if (pCodec == nullptr) {release();return false;}// 分配解码器上下文pCodecCtx = avcodec_alloc_context3(pCodec);if (!pCodecCtx) {release();return false;}pCodecCtx->codec_type = AVMEDIA_TYPE_VIDEO;pCodecCtx->width = mVideoSrcWidth; // 视频宽度pCodecCtx->height = mVideoSrcHeight; // 视频高度pCodecCtx->pix_fmt = AV_PIX_FMT_YUVJ422P; // 或者其他适合的格式pCodecCtx->time_base = { 1, mVideoSrcFps }; // 帧率pCodecCtx->thread_count = 2;// 打开解码器if (avcodec_open2(pCodecCtx, pCodec, nullptr) < 0) {release();return false;}// 分配帧pFrame = av_frame_alloc();pFrameYUYV = av_frame_alloc();if (!pFrame || !pFrameYUYV) {release();return false;}// 分配YUYV帧的缓冲区yuyv_size = av_image_get_buffer_size(AV_PIX_FMT_YUYV422, pCodecCtx->width, pCodecCtx->height, 1);buffer = (uint8_t*)av_malloc(yuyv_size * sizeof(uint8_t));if (!buffer) {release();return false;}av_image_fill_arrays(pFrameYUYV->data, pFrameYUYV->linesize, buffer, AV_PIX_FMT_YUYV422, pCodecCtx->width, pCodecCtx->height, 1);// 创建图像转换上下文sws_ctx = sws_getContext(pCodecCtx->width, pCodecCtx->height, pCodecCtx->pix_fmt,pCodecCtx->width, pCodecCtx->height, AV_PIX_FMT_YUYV422,SWS_BILINEAR, nullptr, nullptr, nullptr);if (!sws_ctx) {release();return false;}
}bool DecodeVideoFrame(uint8_t* data, int size)
{AVPacket* packet = av_packet_alloc();if (!packet) {release();return false;}packet->data = data;packet->size = size;// 发送数据到解码器int ret = avcodec_send_packet(pCodecCtx, packet);if (ret < 0) {release();return false;}// 循环接收解码后的帧while (ret >= 0) {ret = avcodec_receive_frame(pCodecCtx, pFrame);if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF) {break;}else if (ret < 0) {release();return false;}/******** 将解码后数据进行处理 ********/// 转换为YUYV格式if (sws_scale(sws_ctx, pFrame->data, pFrame->linesize, 0, pCodecCtx->height, pFrameYUYV->data, pFrameYUYV->linesize) < 0) {return false;}fwrite(pFrameYUYV->data[0], 1, yuyv_size, out_file);/******** 将解码后数据进行处理 ********/av_frame_unref(pFrame); // 将每次接收的解码帧释放掉,否则会内存泄露}av_packet_unref(packet);av_packet_free(&packet);return true;
}void release()
{if (buffer) {av_free(buffer);buffer = nullptr;}if (pCodecCtx) {avcodec_free_context(&pCodecCtx);pCodecCtx = nullptr;}if (pFormatCtx) {avformat_close_input(&pFormatCtx);pFormatCtx = nullptr;}if (pFrame) {av_frame_free(&pFrame);pFrame = nullptr;}if (pFrameYUYV) {av_frame_free(&pFrameYUYV);pFrameYUYV = nullptr;}if (out_file) {fclose(out_file);}}
http://www.lryc.cn/news/477738.html

相关文章:

  • 前端入门一之CSS知识详解
  • 【JS学习】10. web API-BOM
  • C#实现递归获取所有父级的列表
  • 【深度学习】梯度累加和直接用大的batchsize有什么区别
  • 【Linux】网络相关的命令
  • leetcode哈希表(五)-四数相加II
  • Java学习路线:Maven(一)认识Maven
  • 【深度学习】— 多输入多输出通道、多通道输入的卷积、多输出通道、1×1 卷积层、汇聚层、多通道汇聚层
  • java mapper 的 xml讲解
  • 全面解析:区块链技术及其应用
  • python基础学习笔记
  • 【dvwa靶场:XSS系列】XSS (DOM) 低-中-高级别,通关啦
  • ONLYOFFICE 8.2深度体验:高效协作与卓越性能的完美融合
  • Mac如何将多个pdf文件归并到一个
  • LINUX下的Mysql:Mysql基础
  • 自然语言处理方向学习建议
  • 介绍一下如何生成随机数(c基础)
  • 24-11-1-读书笔记(三十一)-《契诃夫文集》(五)下([俄] 契诃夫 [译] 汝龙)生活乏味但不乏魅力。
  • 从“点”到“面”,热成像防爆手机如何为安全织就“透视网”?
  • 基于vue框架的的奶茶店预约订单系统3fb55(程序+源码+数据库+调试部署+开发环境)系统界面在最后面。
  • 项目实战使用gitee
  • 数据结构--二叉树_链式(下)
  • unity游戏开发之--人物打怪爆材料--拾进背包的实现思路
  • AWTK文件系统适配器更新-支持RT-Thread DFS POSIX接口
  • C#如何快速获取P/Invoke方法签名
  • CqEngine添加联合索引和复合唯一索引
  • 基于matlab的SVPWM逆变器死区补偿算法仿真研究
  • 【网页设计】CSS 定位
  • scala的属性访问权限
  • QGIS:HCMGIS插件