使用ffmpeg保存YUV420p文件

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了使用ffmpeg保存YUV420p文件相关的知识,希望对你有一定的参考价值。


说明

基于 ffmpeg 3.4.2 和 vs2017 开发
YUV420p的存储为先Y,再U,再V。

实现过程

  1. 使用ffmpeg获取frame数据。
  2. 设置sws_getContext为YUV420。
  3. 使用sws_scale转换数据。
  4. 存储数据。

示例代码

代码主要部分为获取frame,转换成YUV格式,YUV数据保存。

// ffmpeg_lesson01.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"
extern "C"

#include "libavcodec/avcodec.h"
#include "libavcodec/version.h"
#include "libavformat/avformat.h"
#include "libavfilter/avfiltergraph.h"
#include "libavfilter/buffersink.h"
#include "libavfilter/buffersrc.h"
#include "libavutil/avutil.h"
#include "libavutil/opt.h"
#include "libavutil/pixdesc.h"
#include "libswscale/swscale.h"
;


int SaveYuv(unsigned char *buf, int wrap, int xsize, int ysize, char *filename)

FILE *f;
int i;

f = fopen(filename, "ab+");
for (i = 0; i<ysize; i++)

fwrite(buf + i * wrap, 1, xsize, f);

fclose(f);
return 1;


int main()



printf("\\n\\n");
printf("正在检查是否是标准C++:");
#if (!defined(__STDC__))
printf("非标准C!\\n");
#elif defined(__STDC_VERSION__)
printf("标准C版本:%ld。", __STDC_VERSION__);
#else
printf("旧的标准C。\\n");
#endif


printf("\\nHello FFmpeg!\\n");
printf("avcodec_version: %d\\n", avcodec_version());
printf("\\n");


#define VIDEO_PATH "D://xiongmao.flv"
char *outputfilename = "d:\\\\test3.yuv";

av_register_all();
//注册所有文件格式和编码器的库。


//下面函数将文件头信息读取到传入的参数中。
AVFormatContext *pFormatContext = NULL;
if (avformat_open_input(&pFormatContext,VIDEO_PATH,NULL,NULL) != 0)
printf("打开输入文件失败!\\n");
return -1;

else
printf("文件打开成功!\\n");

//下面开始检测文件的流信息
if (avformat_find_stream_info(pFormatContext,NULL) < 0)
printf("未发现文件流信息\\n");
return -1;

else
printf("发现文件流信息!\\n");


//打印文件信息
//av_dump_format(pFormatContext,0,VIDEO_PATH,0);

//查找视频流
int videoStream = -1;
for (int i = 0; i < pFormatContext->nb_streams; i++)


if (pFormatContext->streams[i]->codec->codec_type == AVMEDIA_TYPE_VIDEO)
videoStream = i;
printf("发现视频流,索引ID: %d\\n",videoStream);
break;




//查找音频流
int audiostream = -1;

for (int i = 0; i < pFormatContext->nb_streams; i++)


if (pFormatContext->streams[i]->codec->codec_type == AVMEDIA_TYPE_AUDIO)
audioStream = i;
printf("发现音频流,索引ID: %d\\n", audioStream);
break;




//查找到音视频流之后,开始查找音视频对应的解码器。有关编解码器的信息被称为“编码器上下文”
AVCodecContext *pCodecContext;
AVCodec *pCodec;

//查找视频流的编解码信息
pCodecContext = pFormatContext->streams[videoStream]->codec;

pCodec = avcodec_find_decoder(pCodecContext->codec_id);
if (pCodec == NULL)
printf("未找到视频流解码器!\\n");
return -1;
else
printf("找到视频流解码器,ID:%d\\n", pCodecContext->codec_id);


//打开视频编码器
if (avcodec_open2(pCodecContext,pCodec,NULL) < 0)

printf("打开解码器失败!");
return -1;
else
printf("视频解码器打开成功!\\n");



//下面要从视频流中读取一个视频帧并保存成24位RGB色的PPM文件。
AVFrame *pFrame;
//下面开始读取数据
int frameFinished;
AVPacket packet;


//旧的版本使用img_convert就可以。后面改成sws_scale.
//功能更强大。需要详细指出源信息和目的帧信息。
SwsContext *pSwsCtx;
pSwsCtx = sws_getContext(pCodecContext->width,pCodecContext->height,pCodecContext->pix_fmt,pCodecContext->width,pCodecContext->height, AV_PIX_FMT_RGB24, SWS_BICUBIC,NULL,NULL,NULL);



int i = 0;
while (av_read_frame(pFormatContext,&packet) >= 0)

if (packet.stream_index == videoStream) //是否是视频流的数据包

pFrame = av_frame_alloc();

avcodec_decode_video2(pCodecContext, pFrame, &frameFinished, &packet);

if (frameFinished)
SaveYuv(pFrame->data[0], pFrame->linesize[0], pCodecContext->width, pCodecContext->height, outputfilename);
SaveYuv(pFrame->data[1], pFrame->linesize[1], pCodecContext->width / 2, pCodecContext->height / 2, outputfilename);
SaveYuv(pFrame->data[2], pFrame->linesize[2], pCodecContext->width / 2, pCodecContext->height / 2, outputfilename);

av_free(pFrame);


//释放packet
av_free_packet(&packet);


//关闭解码器
avcodec_close(pCodecContext);

//关闭打开的文件
avformat_close_input(&pFormatContext);


return 0;

保存的yuv文件可以用YUVViewer.exe或者yuvplayer.exe播放


以上是关于使用ffmpeg保存YUV420p文件的主要内容,如果未能解决你的问题,请参考以下文章

2023-02-24:请用go语言调用ffmpeg,解码mp4文件并保存为YUV420SP格式文件,采用YUV420P转YUV420SP的方式。

2023-02-25:请用go语言调用ffmpeg,解码mp4文件并保存为YUV420SP格式文件,YUV420P不要转换成YUV420SP。

FFmpeg视频编码 YUV420P编码H264

FFmpeg视频编码 YUV420P编码H264

基于FFmpeg的视频播放器之五:使用SDL2渲染yuv420p

从 YUV420P 到 RGB 的 FFMPEG Api 转换产生奇怪的输出