h264 sps pps详解

2024-04-12 20:58
文章标签 详解 sps h264 pps

本文主要是介绍h264 sps pps详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1、学习路线

step1、初步了解h264,了解h264数据帧分类和识别

step2、h264语法相关算法解析,这里对理解h264的视频帧很重要。通俗的说h264数据定义是由一组Bit组成,但是某个字段不是固定专有几个bit,它是根据其值通过算法转义保存。

step3、sps、pps帧语法定义

2、h264简介

     在H.264/AVC视频编码标准中,整个系统框架被分为了两个层面:视频编码层面(VCL)和网络抽象层面(NAL)。其中,前者负责有效表示视频数据的内容,而后者则负责格式化数据并提供头信息,以保证数据适合各种信道和存储介质上的传输。NAL占一个字节。

    NAL单元(NALU):NAL的基本语法结构,它包含一个字节的头信息和一系列来自VCL的称为原始字节序列载荷(RBSP)的字节流。 数据流是储存在介质上时: 每个NALU 前添加起始码:0x00000001(或者0x000001),用来指示一个 NALU的起始和终止位置。我们平时的每帧数据就是一个NAL单元(SPS与PPS除外)。

    编码器将每个NAL各自独立、完整地放入一个分组,因为分组都有头部,解码器可以方便地检测出NAL的分界,并依次取出NAL进行解码。每个NAL前有一个起始码 0x00 00 01(或者0x00 00 00 01),解码器检测每个起始码,作为一个NAL的起始标识,当检测到下一个起始码时,当前NAL结束。同时H.264规定,当检测到0x000000时,也可以表征当前NAL的结束。那么NAL中数据出现0x000001或0x000000时怎么办?H.264引入了防止竞争机制,如果编码器检测到NAL数据存在0x000001或0x000000时,编码器会在最后个字节前插入一个新的字节0x03,这样:

0x000000->0x00000300
0x000001->0x00000301
0x000002->0x00000302
0x000003->0x00000303
解码器检测到0x000003时,把03抛弃,恢复原始数据(脱壳操作)。解码器在解码时,首先逐个字节读取NAL的数据,统计NAL的长度,然后再开始解码。

NALU头由一个字节组成, 它的语法如下:
      +---------------+
      |0|1|2|3|4|5|6|7|
      +-+-+-+-+-+-+-+-+
      |F|NRI|  Type   |
      +---------------+
F: 1 个比特.
  forbidden_zero_bit. 在 H.264 规范中规定了这一位必须为 0.
NRI: 2 个比特.
  nal_ref_idc. 取 00 ~ 11, 似乎指示这个 NALU 的重要性, 如 00 的 NALU 解码器可以丢弃它而不影响图像的回放. 不过一般情况下不太关心
这个属性.
Type: 5 个比特.
 nal_unit_type. 这个 NALU 单元的类型. 简述如下:
0    没有定义
1     一个非IDR图像的编码条带 (bp帧)
slice_layer_without_partitioning_rbsp( )      
2     编码条带数据分割块A 
slice_data_partition_a_layer_rbsp( )      
3     编码条带数据分割块B 
slice_data_partition_b_layer_rbsp( )      
4     编码条带数据分割块C 
slice_data_partition_c_layer_rbsp( )      
5     IDR图像的编码条带 (i帧)
slice_layer_without_partitioning_rbsp( )      
6     辅助增强信息 (SEI) 
sei_rbsp( )      
7     序列参数集 (sps帧)
seq_parameter_set_rbsp( )      
8     图像参数集 
pic_parameter_set_rbsp( pps帧)      
9     访问单元分隔符 
access_unit_delimiter_rbsp( )      
10     序列结尾 
end_of_seq_rbsp( )      
11     流结尾 
end_of_stream_rbsp( )      
12     填充数据 
filler_data_rbsp( )      
13     序列参数集扩展 
seq_parameter_set_extension_rbsp( )      
14...18     保留      
19     未分割的辅助编码图像的编码条带 
slice_layer_without_partitioning_rbsp( )      
20...23     保留      
24    STAP-A   单一时间的组合包
25    STAP-B   单一时间的组合包
26    MTAP16   多个时间的组合包
27    MTAP24   多个时间的组合包
28    FU-A     分片的单元
29    FU-B     分片的单元
30-31 没有定义

当遇到 00 00 00 01 67表示sps帧

当遇到 00 00 00 01 68 表示pps帧

3、h264语法相关算法解析 

1、无符号指数哥伦布熵编码

1.1 编码过程

1、将待编码的数加1转换为最小的二进制序列(假设一共M位); 
2、此二进制序列前面补充M-1个0; 
3、enjoy!

1.1.1 示例

对 4 进行无符号指数哥伦布熵编码 
1、将4加1(为5)转换为最小的二进制序列即 101 (此是M=3) 
2、此二进制序列前面补充M-1即两个0 
3、得出的4的无符号指数哥伦布熵编码的序列为 00101

1.2 解码过程

1、获取二进制序列开头连续的N个0 
2、读取之后的N+1位的值,假设为X 
3、X-1获取解码后的值

1.2.1 示例

如对 00101进行无符号指数哥伦布熵解码 
1、获取开头连续的N个0, 此时N = 2 
2、再向后读取N+1位的值,即 101,为5 
3、 5 - 1 =4 获取其解码后码值,enjoy!

1.3 其他

注意0的无符号指数哥伦布熵编码的二进制序列为 1

2 有符号指数哥伦布熵编码

2.1 编码过程

1、将待编码的数的绝对值转换为最小的二进制序列(假设一共M位) 
2、在此二进制序列后补充一位符号位0表示正,1表示负 
3、在此二进制序列前补充M个0 
4、enjoy

2.1.1 示例1

如对4进行有符号指数哥伦布熵编码 
1、4的绝对值转为最小二进制序列,即 100 (此时M = 3) 
2、后面补充符号位,0 即 1000 
3、前面补充M个0, 即 0001000 
4、enjoy

2.1.2 示例2

如对-15进行有符号指数哥伦布熵编码 
1、-7的绝对值转为最小二进制序列,即 1111 (此时M = 4) 
2、后面补充符号位,1,即 11111 
3、前面补充M个0,即 000011111 
4、enjoy

2.2 解码过程

1、获取二进制序列开头连续的N个0 
2、读取之后的N位的值,假设为X 
3、获取最后1位符号位 
4、获取解码后码值

2.2.1 示例1

如对二进制序列 0001000 进行有符号指数哥伦布熵解码 
1、获取开头连续的N个0, 此时N = 3 
2、再获取N为数值,即 100 即为4 
3、获取最后的符号位,0,即为正值 
4、故此序列解码后的码值为4

2.2.2 示例2

如对二进制序列 000011111 进行有符号指数哥伦布熵解码 
1、获取开头连续的N个0, 此时N = 4 
2、再获取N为数值,即 1111 即为15 
3、获取最后的符号位,1,即为负值 
4、故此序列解码后的码值为-15

 

4、sps语法

 

5、pps语法

 

 



转载自:http://blog.csdn.net/yuanbinquan/article/details/60148345


这篇关于h264 sps pps详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/898219

相关文章

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

K8S(Kubernetes)开源的容器编排平台安装步骤详解

K8S(Kubernetes)是一个开源的容器编排平台,用于自动化部署、扩展和管理容器化应用程序。以下是K8S容器编排平台的安装步骤、使用方式及特点的概述: 安装步骤: 安装Docker:K8S需要基于Docker来运行容器化应用程序。首先要在所有节点上安装Docker引擎。 安装Kubernetes Master:在集群中选择一台主机作为Master节点,安装K8S的控制平面组件,如AP

嵌入式Openharmony系统构建与启动详解

大家好,今天主要给大家分享一下,如何构建Openharmony子系统以及系统的启动过程分解。 第一:OpenHarmony系统构建      首先熟悉一下,构建系统是一种自动化处理工具的集合,通过将源代码文件进行一系列处理,最终生成和用户可以使用的目标文件。这里的目标文件包括静态链接库文件、动态链接库文件、可执行文件、脚本文件、配置文件等。      我们在编写hellowor

LabVIEW FIFO详解

在LabVIEW的FPGA开发中,FIFO(先入先出队列)是常用的数据传输机制。通过配置FIFO的属性,工程师可以在FPGA和主机之间,或不同FPGA VIs之间进行高效的数据传输。根据具体需求,FIFO有多种类型与实现方式,包括目标范围内FIFO(Target-Scoped)、DMA FIFO以及点对点流(Peer-to-Peer)。 FIFO类型 **目标范围FIFO(Target-Sc

019、JOptionPane类的常用静态方法详解

目录 JOptionPane类的常用静态方法详解 1. showInputDialog()方法 1.1基本用法 1.2带有默认值的输入框 1.3带有选项的输入对话框 1.4自定义图标的输入对话框 2. showConfirmDialog()方法 2.1基本用法 2.2自定义按钮和图标 2.3带有自定义组件的确认对话框 3. showMessageDialog()方法 3.1

脏页的标记方式详解

脏页的标记方式 一、引言 在数据库系统中,脏页是指那些被修改过但还未写入磁盘的数据页。为了有效地管理这些脏页并确保数据的一致性,数据库需要对脏页进行标记。了解脏页的标记方式对于理解数据库的内部工作机制和优化性能至关重要。 二、脏页产生的过程 当数据库中的数据被修改时,这些修改首先会在内存中的缓冲池(Buffer Pool)中进行。例如,执行一条 UPDATE 语句修改了某一行数据,对应的缓

OmniGlue论文详解(特征匹配)

OmniGlue论文详解(特征匹配) 摘要1. 引言2. 相关工作2.1. 广义局部特征匹配2.2. 稀疏可学习匹配2.3. 半稠密可学习匹配2.4. 与其他图像表示匹配 3. OmniGlue3.1. 模型概述3.2. OmniGlue 细节3.2.1. 特征提取3.2.2. 利用DINOv2构建图形。3.2.3. 信息传播与新的指导3.2.4. 匹配层和损失函数3.2.5. 与Super

web群集--nginx配置文件location匹配符的优先级顺序详解及验证

文章目录 前言优先级顺序优先级顺序(详解)1. 精确匹配(Exact Match)2. 正则表达式匹配(Regex Match)3. 前缀匹配(Prefix Match) 匹配规则的综合应用验证优先级 前言 location的作用 在 NGINX 中,location 指令用于定义如何处理特定的请求 URI。由于网站往往需要不同的处理方式来适应各种请求,NGINX 提供了多种匹