转自: https://www.freehacker.cn/media/codec-h264/
H.264,又称为 MPEG-4 第10部分,高级视频编码(英语:MPEG-4 Part 10, Advanced Video Coding,缩写为 MPEG-4 AVC)是一种面向块的基于运动补偿的视频编码标准 。
对于视频序列样本来说,使用 H.264 编码器能够比使用有运动补偿的 MPEG-4 编码器降低50%的比特率(bps)。在没有运动补偿的情况下,H.264 编码器的效率至少比 MPEG-4 编码器高3倍,比 M-JPEG 编码器高6倍。除了能够显著的提高编码效率外,H.264的优点还包括:提供高质量的图像、容错能力强、网络适应性强、低时延等。
有关 H264 详细的介绍可以参考
维基百科:H.264/MPEG-4 AVC,中文地址/英文地址
H264相关标准,中英对照版本见文献。
视频编码的核心思想是是去除冗余信息,冗余信息包括以下几个方面:
H.264压缩方法如下:
要想了解帧与场的概念,需先掌握逐行扫描和隔行扫描的区别。在英语中,逐行扫描为Progressive Scanning,隔行扫描为Interlace Scanning。
逐行比较好理解,假设一帧图像的大小是704x576,那么逐行的话就是576行。隔行图像,是早期电视信号中引入的概念,把一帧图像分为上下两场,两场图像在时间上具有先后,但传输时同时传送到显示端,显示端在显示按各自的时间分开进行显示。该方式主要是利用了人眼的余辉效应,通过隔行显示,提高了显示的流畅性。
场(Field)和帧(Frame)的概念一目了然:
对于一个具体的序列参数集来说,编码场的宽度和编码帧的宽度是相同的,而编码场的高度是是编码帧的高度的一半。
H.264多了个图像组成一个视频系列(GOP),每个图像由一帧(Frame)组成。帧又可以分成一个或几个片(Slice)。片由宏块(Macro Block)组成,一帧中每个片的宏块数不一定相同。每个宏块由一个16×16的亮度数组和两个8×8的色差数组组成。一个720×480分辨率的图像可以被划分为1350(45×30)个宏块。
宏块是编码处理的基本单元。一个帧是可以分割成多个片来编码的,构成I_Slice、P_Slice、B_Slice、SP_Slice和SI_Slice,而一个片编码之后被打包进一个NAL单元。编码片相互间独立,这样做的目的是为了限制误码的扩散和传输。
I_Slice、P_Slice和B_Slice对应于I帧、P帧和B帧,三者之区别如下:
根据H.264的不同类别,编码器会使用不同类型的帧,例如I帧、P帧和B帧。
I帧(帧内编码帧)是一种自带全部信息的独立帧,无需参考其它图像便可独立进行解码。
视频序列中的第一个帧始终都是I帧。如果所传输的比特流遭到破坏,则需要将I帧用作新查看器的起始点或重新同步点。I帧可以用来实现快进、快退以及其它随机访问功能。如果新的客户端将参与查看视频流,编码器将以相同的时间间隔或者根据要求自动插入I帧。I帧的缺点在于它们会占用更多的数据位,但从另一方面看,I帧不会产生可觉察的模糊现象。
P帧(帧间预测编码帧)需要参考前面的I帧和/或P帧的不同部分才能进行编码。
与I帧相比,P帧通常占用更少的数据位,但其缺点是,由于P帧对前面的P和I参考帧有着复杂的依赖性,因此对传输错误非常敏感。
B帧(双向预测编码帧)需要同时以前面的帧和后面的帧作为参考帧。
下图为带有I帧、B帧和P帧的典型视频序列。P帧只需要参考前面的I帧或P帧,而B帧则需要同时参考前面和后面的I帧或P 帧。
当视频解码器逐个帧地对比特流进行解码以便重构视频时,必须始终从I帧开始解码。如果使用了P帧和B帧,则必须与参考帧一起解码。在H.264基准类中,仅使用I帧和P帧。由于基准类没有使用B帧,所以可以实现低延时,因此是网络摄像机和视频编码器的理想选择。
在H.264中,将通过新的高级帧内预测方法对I帧进行编码。这种方法通过对帧中每个宏块内较小的像素块进行连续预测,通过一组不同方向上的相邻块来预测当前块,可以大大减少I帧所占的数据位并保持较高的质量。这一点可通过在与进行帧内编码的新4×4像素块相邻接的前几个编码像素中,寻找匹配的像素来实现。通过重复利用已编码的像素值,可以极大地减少需要编码的位数。
帧内预测编码的是预测块和实际块差值,这种方法能够有效减少平滑背景小的空间冗余。
H.264 通过差分编码来减少视频数据量,大多数视频压缩标准都采用这种方法:在差分编码中,会将一个帧与参考帧(即前面的 I 帧或 P 帧)进行对比,然后只对那些相对于参考帧来说发生了变化的像素进行编码。通过这种方法,可以降低需要进行编码和发送的像素值。
对差分编码(包括H.264在内的大多数视频压缩标准都采用这种方法)来说,只有第一个图像(I帧)是将全帧图像信息进行编码。在后面的两个图像(P帧)中,其静态部分(即房子)将参考第一个图像,而仅对运动部分(即正在跑步的人)使用运动矢量进行编码,从而减少发送和存储的信息量。
然而,如果视频中存在大量物体运动的话,差分编码将无法显著减少数据量。这时,可以采用基于块的运动补偿技术。基于块的运动补偿考虑到视频序列中构成新帧的大量信息都可以在前面的帧中找到,但可能会在不同的位置上。所以,这种技术将一个帧分为一系列的宏块。然后,通过在参考帧中查找匹配块的方式,逐块地构建或者”预测”一个新帧(例如P帧)。如果发现匹配的块,编码器只需要对参考帧中发现匹配块的位置进行编码。与对块的实际内容进行编码相比,只对运动矢量进行编码可以减少所占用的数据位。
为了提高编码效率,宏块被分割成更小的子块,下图给出了 H.264、MPEG-4和MPEG-2运动补偿技术特性对比:
同时,由于运动是个持续的过程,拍摄运动图像时可能会出现偏移和遮挡,拍摄角度会不停的来回切换,H.264引入了多参考帧的概念。
从帧间预测和帧内预测得到的结果需要从空域转换成频域,H.264/MPEG-4 AVC采用4×4 DCT-like整数变换。相对于MPEG-2和MPEG-4基于浮点系数的8×8 DCT变换,整数系数能够消除浮点系数固有的舍入误差,进而消除浮点系数引起的drifting artifacts。同时,H.264/MPEG-4 AVC较小的块大小也减少了阻塞和ringing artifacts。
H.264协议中的变化方式主要有三种:4×4残差变化(整数DCT变换)、16×16帧内模式下4×4亮度直流系数变化(离散哈达玛变化)、2×2色度直流系数变化(离散哈达玛变化)。
从整数变化阶段得到的系数需要被量化,用以减少整数系数的整体精度,并趋向于消除高频系数,保持感知质量。量化的原则是在不降低视觉效果的前提下尽量减少图像编码长度,减少视觉恢复中不必要的信息。
一般的标量量化器原理如下:
Zij=round(Yij/Qstep)
其中,Yij表示一个宏块经过整数DCT转换后的系数,Zij是输出的量化系数,Qstep是量化步长。量化步长决定了量化器的编码压缩率以及图像精度。如果量化步长较大,则量化值Zij动态范围较小,其相应的编码长度较小,但反量化会损失较多的图像细节信息;如果量化步长较小,则量化值Zij动态范围较大,其相应的编码长度较大,但图像损失较少。H.264编码器根据图像值实际动态范围自动改变量化步长值,在编码长度和图像精度之间折衷,达到整体最佳效果。
在H.264中,使用量化参数QP(Quantization Parameter)来标识量化步长的序号,亮度编码的量化步长Qstep共有52个值,QP为0-51;色度编码的量化步长Qstep共有39个值,QP为0-39。QP取最小值0 时,表示量化最精细;相反,QP取最大值51/39时,表示量化是最粗糙的。亮度编码的QP每增加6,量化步长Qstep增加一倍。亮度编码的量化步长信息如下表:
QP | Qstep | QP | Qstep | QP | Qstep | QP | Qstep | QP | Qstep |
---|---|---|---|---|---|---|---|---|---|
0 | 0.625 | 13 | 2.25 | 22 | 8 | 33 | 28 | 44 | 104 |
1 | 0.6875 | 12 | 2.5 | 23 | 9 | 34 | 32 | 45 | 112 |
2 | 0.8125 | 13 | 2.75 | 24 | 10 | 35 | 36 | 46 | 128 |
3 | 0.875 | 14 | 3.25 | 25 | 11 | 36 | 40 | 47 | 144 |
4 | 1 | 15 | 3.5 | 26 | 13 | 37 | 44 | 48 | 160 |
5 | 1.125 | 16 | 4 | 27 | 14 | 38 | 52 | 49 | 176 |
6 | 1.25 | 17 | 4.5 | 28 | 16 | 39 | 56 | 50 | 208 |
7 | 1.375 | 18 | 5 | 29 | 18 | 40 | 64 | 51 | 224 |
8 | 1.625 | 19 | 5.5 | 30 | 20 | 41 | 72 | ||
9 | 1.75 | 20 | 6.5 | 31 | 22 | 42 | 80 | ||
10 | 2 | 21 | 7 | 32 | 26 | 43 | 88 |
H.264 / MPEG-4 AVC定义了一种去块滤波器(de-blocking filter ),其对16×16宏块和4×4块边界进行操作。 在宏块的情况下,滤波器旨在去除可能由具有不同估计类型(例如,运动与帧内估计)的相邻宏块和/或不同的量化尺度产生的伪像。 在块的情况下,滤波器旨在去除可能由变换/量化引起的伪像和相邻块之间的运动矢量差异。 循环滤波器通常使用内容自适应非线性滤波器来修改宏块/块边界的任一侧上的两个像素。
在进行熵编码之前,必须对4x4量化系数进行序列化。 根据这些系数是否是原始的运动估计或帧内估计,选择不同的扫描模式来创建串行化流。 扫描模式将系数从低频到高频排列。 然后,由于较高频率的量化系数趋向于零,所以使用游程长度编码来分组尾随零,导致更有效的熵编码。
熵编码将表示运动矢量、量化系数和宏块头的符号映射到实际位中,熵编码通过将较少数量的比特分配给频繁使用的符号和较大数量的比特到较不频繁使用的符号来提高编码效率。
下列表格描述了两种主要类型的熵编码:Variable Length Coding (VLC)和Context Adaptive Binary Arithmetic Coding (CABAC)。
H.264 标准的一个重要方面是通过级别(Level)和档次(Profile)中提供的功能,以最佳的方式支持常见应用和通用格式。Profile 是对视频压缩特性的描述(CABAC 呀、颜色采样数等等),Level 是对视频本身特性的描述(码率、分辨率、帧率)。简单来说,Profile 越高,就说明采用了越高级的压缩特性。Level 越高,视频的码率、分辨率、帧率越高。
H.264 从低到高划分了很多 Profile 和 Level,在维基百科英文版中能够看到详细的表格,在本节末尾会贴出相关表格内容。
H.264 支持四个 Profile,分别为:
制定H.264的主要目标有两个:
视频编码层进行视频数据压缩、解压缩操作,而网络抽象层专门为视频编码信息提供头文件信息,安排格式以方便网络传输和介质存储。VCL数据即编码处理的输出,它表示被压缩编码后的视频数据序列。在VCL数据传输或存储之前,这些编码的VCL数据,先被映射或封装进NAL单元中。每个NAL单元包含一个原始字节序列负载(RBSP,Raw Byte Sequence Payload)、一组对应于视频编码数据的NAL头信息。其具体结构如下图:
在H.264编码过程中,存在三种不同的数据形式:
加上仿校验字节的原因是:EBSP被封装为NALU时,需要为其添加开始前缀,如果该NALU对应的Slice为一帧的开始,则用’0x000000001’,否则使用’0x000001’。
为了使NALU主体中不包括与开始码相冲突的,在编码时,每遇到两个字节连续为0,就插入一个字节的0x03。解码时将0x03去掉。也称为脱壳操作。
在H.264中图像以序列为单位进行组织,一个序列是一段图像编码后的数据流,以IDR帧开始,到下一个IDR帧结束,中间包含若干访问单元(Access Unit)。从宏观上来说,SPS、PPS、IDR 帧(包含一个或多个I-Slice)、P 帧(包含一个或多个P-Slice )、B 帧(包含一个或多个B-Slice )共同构成典型的H.264码流结构。
SPS,Sequence Parameter Sets,序列参数集,是H.264码流序列的第一个NALU,PPS,Picture Parameter Set,图像参数集,是H.264码流序列的第二个NALU。
SPS和PPS中包含了初始化H.264解码器所需要的信息参数,包括编码所用的profile、level、图像的宽和高、deblock滤波器等。SPS语法单元存放一个视频序列共同特征,而PPS语法单元各个图像的典型特征。SPS和PPS都各自对应于一个NALU。
只有视频序列之间才能切换SPS,即只有IDR帧的第一个slice才可以切换SPS;只有图像之间才能切换PPS,即只有每帧图像的第一个slice才能切换PPS。
IDR,Instantaneous Decoder Refresh,即时解码器刷新,是H.264码流序列的第三个NALU。
IDR帧都是I帧,但是I帧并不一定是IDR帧。H.264 引入 IDR 图像是为了解码的重同步,当解码器解码到 IDR 图像时,立即将参考帧队列清空,将已解码的数据全部输出或抛弃,重新查找参数集,开始一个新的序列。这样,如果前一个序列出现重大错误,在这里可以获得重新同步的机会。IDR图像之后的图像永远不会使用IDR之前的图像的数据来解码。
在IDR帧之后可能存在I帧、P帧和B帧,这个具体看使用的是哪种Profile。
NALU,Network Abstract Layer Unit,是H.264的最高抽象层,H.264的所有语法结构最终都被封装成NALU,同时会加入一些网络相关信息。码流中的NALU单元必须定义合适的分隔符,否则无法区分。H.264 视频压缩标准的附录B采用前缀码“00 00 01”/“00 00 00 01”作为NALU的分隔符,可以通过搜索前缀码“00 00 01”/“00 00 00 01”来识别一个NALU。
H.264视频流是以NAL单元传送的,但在一个NAL单元里面,可能既存放I-Slice(P-Slice或B-Slice),也可能存放图像的其他信息,比如SPS、PPS。根据H.264语法结构,根据的NAL类型可以分为:
NAL_SLICE = 1 SLICE非IDR不分割
NAL_SLICE_DPA = 2 SLICE数据分割块A
NAL_SLICE_DPB = 3 SLICE数据分割快B
NAL_SLICE_DPC = 4 SLICE数据分割块C
NAL_SLICE_IDR = 5 SLICE关键帧
NAL_SEI = 6 补充增强信息单元帧
NAL_SPS = 7 序列参数集
NAL_PPS = 8 图像参数集
NAL_AUD = 9 分界符
NAL_EOSEQ = 10 序列结束
NAL_EOSTREAM = 11 码流结束
NAL_FILLER = 12 填充
= 13
= ...
= 23 13~23保留
= 24
= ...
= 31 24~32不保留,RTP打包时会用到
将数据块分割为A、B、C,主要目的是为了对重要程度不同的数据进行不同程度的保护。其中24~32类型是不保留的,其在RTP打包时会用到:
类型取值 | 对应类型 | 功能 |
---|---|---|
24 | STAP-A | Single-time aggregation packet |
25 | STAP-B | Single-time aggregation packet |
26 | MTAP16 | Multi-time aggregation packet |
27 | MTAP24 | Multi-time aggregation packet |
28 | FU-A | Fragmentation unit |
29 | FU-B | Fragmentation unit |
30-31 | undefined |
对于比较大的NALU单元,是无法一次通过RTP发送的(RTP的MTU为1500),所以必须要拆包,将较大的NALU拆分为FU-A包。这里面有拆包和解包两个概念:
原始的NAL头的前三位为FU indicator的前三位,原始的NAL头的后五位为FU header的后五位,
FU indicator与FU header的剩余位数根据实际情况决定。
还原后的NAL头的八位是由FU indicator的前三位加FU header的后五位组成,
即:nal_unit_type = (fu_indicator & 0xe0) | (fu_header & 0x1f)
H.264 的开源实现包括:OpenH264 和 x264。OpenH264 是思科实现的开源 H.264 编码,OpenH264 是思科实现的开源 H.264 编码。两者的对比如下:
目前能与H.264相提并论的编码算法包括:
HEVC/H.265:开源实现包括 libde265 和 x265,需专利费;
VP8:开源实现为 libvpx,无专利费;
VP9:开源实现为 ibvpx,无专利费。
几种编码方案中,HEVC 对 VP9 和 H.264 在码率上有较大优势,在相同 PSNR 下分别节省了 48.3% 和 75.8%。H.264 在编码时间上有巨大优势,对比 VP9 和 HEVC(H.265) ,HEVC 是 VP9 的 6 倍,VP9 是 H.264 的将近 40 倍。
在H.264之前,还有一些前辈算法,包括: