1、播放器框架
1.1常用音视频术语
-
容器/文件(Conainer/File):即特定格式的多媒体文件,比如mp4、flv、mkv等。
-
媒体流(Stream):表示时间轴上的一段连续数据,如一段声音数据、一段视频数据或一段字幕数据,可以是压缩的,也可以是非压缩的,压缩的数据需要关联特定的编解码器(有些码流音频他是纯PCM)。
-
数据帧/数据包(Frame/Packet):通常,一个媒体流是由大量的数据帧组成的,对于压缩数据,帧对应着编解码器的最小处理单元,分属于不同媒体流的数据帧交错存储于容器之中。
-
编解码器:编解码器是以帧为单位实现压缩数据和原始数据之间的相互转换的。
2、常用概念
-
复用器
-
编解码器
3、FFmpeg库简介
FFMPEG有8个常用库:
-
AVUtil:核心工具库,下面的许多其他模块都会依赖该库做一些基本的音视频处理操作。
-
AVFormat:文件格式和协议库,该模块是最重要的模块之一,封装了Protocol层和Demuxer、Muxer层,使得协议和格式对于开发者来说是透明的。
-
AVCodec:编解码库,封装了Codec层,但是有一些Codec是具备自己的License的,FFmpeg是不会默认添加像libx264、FDK-AAC等库的,但是FFmpeg就像一个平台一样,可以将其他的第三方的Codec以插件的方式添加进来,然后为开发者提供统一的接口。
-
AVFilter:音视频滤镜库,该模块提供了包括音频特效和视频特效的处理,在使用FFmpeg的API进行编解码的过程中,直接使用该模块为音视频数据做特效处理是非常方便同时也非常高效的一种方式。
-
AVDevice:输入输出设备库,比如,需要编译出播放声音或者视频的工具ffplay,就需要确保该模块是打开的,同时也需要SDL的预先编译,因为该设备模块播放声音与播放视频使用的都是SDL库。
-
SwrRessample:该模块可用于音频重采样,可以对数字音频进行声道数、数据格式、采样率等多种基本信息的转换。
-
SWScale:该模块是将图像进行格式转换的模块,比如,可以将YUV的数据转换为RGB的数据,缩放尺寸由1280720变为800480。
-
PostProc:该模块可用于进行后期处理,当我们使用AVFilter的时候需要打开该模块的开关,因为Filter中会使用到该模块的一些基础函数。
-
av_register_all():注册所有组件,4.0已经弃用
-
avdevice_register_all()对设备进行注册,比如V4L2等。
-
avformat_network_init();初始化网络库以及网络加密协议相关的库(比如openssl)
封装格式相关
-
avformat_alloc_context();负责申请一个AVFormatContext结构的内存,并进行简单初始化
-
avformat_free_context();释放该结构里的所有东西以及该结构本身
-
avformat_close_input();关闭解复用器。关闭后就不再需要使用avformat_free_context 进行释放。
-
avformat_open_input();打开输入视频文件
-
avformat_find_stream_info():获取音视频文件信息
-
av_read_frame(); 读取音视频包
-
avformat_seek_file(); 定位文件
-
av_seek_frame():定位文件
解码器相关
-
avcodec_alloc_context3(): 分配解码器上下文
-
avcodec_find_decoder():根据ID查找解码器
-
avcodec_find_decoder_by_name():根据解码器名字
-
avcodec_open2(): 打开编解码器
-
avcodec_decode_video2():解码一帧视频数据
-
avcodec_decode_audio4():解码一帧音频数据
-
avcodec_send_packet(): 发送编码数据包
-
avcodec_receive_frame(): 接收解码后数据
-
avcodec_free_context():释放解码器上下文,包含了avcodec_close()
-
avcodec_close():关闭解码器
FFmpeg3.x组件注册方式
我们使用ffmpeg,首先要执行av_register_all,把全局的解码器、编码器等结构体注册到各自全局的对象链表里,以便后面查找调用。
FFmpeg4.x组件注册方式
FFmpeg4.0.2组件注册方式
4、FFmpeg常用结构体简介
-
AVFormatContext 封装格式上下文结构体,也是统领全局的结构体,保存了视频文件封装格式相关信息。
-
AVInputFormat demuxer 每种封装格式(例如FLV, MKV, MP4, AVI)对应一个该结构体。
-
AVOutputFormat muxer
-
AVStream 视频文件中每个视频(音频)流对应一个该结构体。
-
AVCodecContext 编解码器上下文结构体,保存了视频(音频)编解码相关信息。
-
AVCodec 每种视频(音频)编解码器(例如H.264解码器)对应一个该结构体。
-
AVPacket 存储一帧压缩编码数据。
-
AVFrame 存储一帧解码后像素(采样)数据。
4.1FFmpeg数据结构之间的关系
AVFormatContext和AVInputFormat之间的关系
AVFormatContext 封装格式上下文结构体
struct AVInputFormat *iformat;
复制代码
AVInputFormat 每种封装格式(例如FLV, MKV, MP4)
int (*read_header)(struct AVFormatContext * );
int (*read_packet)(struct AVFormatContext *, AVPacket *pkt);
复制代码
int avformat_open_input(AVFormatContext **ps, const char *filename,AVInputFormat *fmt, AVDictionary **options)
复制代码
AVCodecContext和AVCodec之间的关系
AVCodecContext 编码器上下文结构体
struct AVCodec *codec;
复制代码
AVCodec 每种视频(音频)编解码器
int (*decode)(AVCodecContext *, void *outdata, int *outdata_size,
AVPacket *avpkt);
int (*encode2)(AVCodecContext *avctx, AVPacket *avpkt, const AVFrame *frame, int *got_packet_ptr);
复制代码
4.2区分不同的码流
AVMEDIA_TYPE_VIDEO视频流
video_index = av_find_best_stream(ic, AVMEDIA_TYPE_VIDEO, -1,-1, NULL, 0)
复制代码
- AVMEDIA_TYPE_AUDIO音频流
audio_index = av_find_best_stream(ic, AVMEDIA_TYPE_AUDIO, -1,-1, NULL, 0)
复制代码
AVPacket 里面也有一个index的字段
4.3数据结构分析
AVFormatContext
-
iformat:输入媒体的AVInputFormat,比如指向AVInputFormat ff_flv_demuxer
-
nb_streams:输入媒体的AVStream 个数
-
streams:输入媒体的AVStream []数组
-
duration:输入媒体的时长(以微秒为单位),计算方式可以参考av_dump_format()函数。
-
bit_rate:输入媒体的码率
AVInputFormat
AVStream
AVCodecParameters
AVCodecContext
AVCodec
AVCodecContext
AVCodec
AVPacket
-
pts:显示时间戳
-
dts:解码时间戳
-
data:压缩编码数据
-
size:压缩编码数据大小
-
pos:数据的偏移地址
-
stream_index:所属的AVStream
AVFrame
-
data:解码后的图像像素数据(音频采样数据)
-
linesize:对视频来说是图像中一行像素的大小;对音频来说是整个音频帧的大小
-
width, height:图像的宽高(只针对视频)
-
key_frame:是否为关键帧(只针对视频) 。
-
pict_type:帧类型(只针对视频) 。例如I, P, B
-
sample_rate:音频采样率(只针对音频)
-
nb_samples:音频每通道采样数(只针对音频)
-
pts:显示时间戳
原文链接:C/C++音视频高级开发 FFmpeg编程入门 - 掘金
★文末名片可以免费领取音视频开发学习资料,内容包括(FFmpeg ,webRTC ,rtmp ,hls ,rtsp ,ffplay ,srs)以及音视频学习路线图等等。
见下方!↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)