H264 获取SPS与PPS

2024-06-19 16:18
文章标签 获取 sps h264 pps

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

转载地址

H264 获取SPS与PPS

在用Android手机进行h264硬编码的时候如果要进行视频流的实时传输与播放,就需要知道视频流的Sequence Parameter Sets (SPS) 和Picture Parameter Set (PPS)。

今天算是看明白如何获取SPS和PPS,在这里记录下来,希望有需要的朋友可以在这里获取到一些些的帮助。

首先说一下大前提,我设置的视频录制参数为:

mMediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);

mMediaRecorder.setVideoEncoder(MediaRecorder.VideoEncoder.H264);

为了让大家更加明白,我先贴出avcC的数据结构:

    aligned(8) class AVCDecoderConfigurationRecord {  unsigned int(8) configurationVersion = 1;  unsigned int(8) AVCProfileIndication;  unsigned int(8) profile_compatibility;  unsigned int(8) AVCLevelIndication;  bit(6) reserved = '111111'b;  unsigned int(2) lengthSizeMinusOne;  bit(3) reserved = '111'b;  unsigned int(5) numOfSequenceParameterSets;  for (i=0; i< numOfSequenceParameterSets; i++) {  unsigned int(16) sequenceParameterSetLength ;  bit(8*sequenceParameterSetLength) sequenceParameterSetNALUnit;  }  unsigned int(8) numOfPictureParameterSets;  for (i=0; i< numOfPictureParameterSets; i++) {  unsigned int(16) pictureParameterSetLength;  bit(8*pictureParameterSetLength) pictureParameterSetNALUnit;  }  }  

ok,数据结构贴出来了,我再贴出录制的3gp输出类型的h264码流片断。

这里写图片描述

阴影部分就是avcC的全部数据了。

其中: 0x61 0x76 0x63 0x43 就是字符avcC

0x01 是configurationVersion

0x42 是AVCProfileIndication

0x00 是profile_compatibility

0x1F是AVCLevelIndication

0xFF 是6bit的reserved 和2bit的lengthSizeMinusOne

0xE1 是3bit的reserved 和5bit的numOfSequenceParameterSets

0x00 0x09是sps的长度为9个字节。

故SPS的内容为接下来的9个字节:67 42 00 1f e9 02 c1 2c 80

接下来的:01为numOfPictureParameterSets

0x00和0x04是pps的长度为4个字节。

故PPS的内容为接下来的4个字节:68 ce 06 f2

通过这段数据片断,就可以获取到SPS和PPS了。

下面我将贴上用java代码获取输出格式为3gp的h264码流的SPS与PPS代码:

    package cn.edu.xmu.zgy;  import java.io.File;  import java.io.FileInputStream;  import java.io.IOException;  public class ObtainSPSAndPPS {  public void getSPSAndPPS(String fileName) throws IOException {  File file = new File(fileName);  FileInputStream fis = new FileInputStream(file);  int fileLength = (int) file.length();  byte[] fileData = new byte[fileLength];  fis.read(fileData);  // 'a'=0x61, 'v'=0x76, 'c'=0x63, 'C'=0x43  byte[] avcC = new byte[] { 0x61, 0x76, 0x63, 0x43 };  // avcC的起始位置  int avcRecord = 0;  for (int ix = 0; ix < fileLength; ++ix) {  if (fileData[ix] == avcC[0] && fileData[ix + 1] == avcC[1]  && fileData[ix + 2] == avcC[2]  && fileData[ix + 3] == avcC[3]) {  // 找到avcC,则记录avcRecord起始位置,然后退出循环。  avcRecord = ix + 4;  break;  }  }  if (0 == avcRecord) {  System.out.println("没有找到avcC,请检查文件格式是否正确");  return;  }  // 加6的目的是为了跳过  // (1)8字节的 configurationVersion  // (2)8字节的 AVCProfileIndication  // (3)8字节的 profile_compatibility  // (4)8 字节的 AVCLevelIndication  // (5)6 bit 的 reserved  // (6)2 bit 的 lengthSizeMinusOne  // (7)3 bit 的 reserved  // (8)5 bit 的numOfSequenceParameterSets  // 共6个字节,然后到达sequenceParameterSetLength的位置  int spsStartPos = avcRecord + 6;  byte[] spsbt = new byte[] { fileData[spsStartPos],  fileData[spsStartPos + 1] };  int spsLength = bytes2Int(spsbt);  byte[] SPS = new byte[spsLength];  // 跳过2个字节的 sequenceParameterSetLength  spsStartPos += 2;  System.arraycopy(fileData, spsStartPos, SPS, 0, spsLength);  printResult("SPS", SPS, spsLength);  // 底下部分为获取PPS  // spsStartPos + spsLength 可以跳到pps位置  // 再加1的目的是跳过1字节的 numOfPictureParameterSets  int ppsStartPos = spsStartPos + spsLength + 1;  byte[] ppsbt = new byte[] { fileData[ppsStartPos],  fileData[ppsStartPos + 1] };  int ppsLength = bytes2Int(ppsbt);  byte[] PPS = new byte[ppsLength];  ppsStartPos += 2;  System.arraycopy(fileData, ppsStartPos, PPS, 0, ppsLength);  printResult("PPS", PPS, ppsLength);  }  private int bytes2Int(byte[] bt) {  int ret = bt[0];  ret <<= 8;  ret |= bt[1];  return ret;  }  private void printResult(String type, byte[] bt, int len) {  System.out.println(type + "长度为:" + len);  String cont = type + "的内容为:";  System.out.print(cont);  for (int ix = 0; ix < len; ++ix) {  System.out.printf("%02x ", bt[ix]);  }  System.out.println("\n----------");  }  public static void main(String[] args) throws IOException {  new ObtainSPSAndPPS().getSPSAndPPS("c:\\zgy.h264");  }  }  

运行结果如下:

    SPS长度为:9  SPS的内容为:67 42 00 1f e9 02 c1 2c 80   ----------  
    PPS长度为:4  PPS的内容为:68 ce 06 f2   ----------  

这篇关于H264 获取SPS与PPS的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Android Environment 获取的路径问题

1. 以获取 /System 路径为例 /*** Return root of the "system" partition holding the core Android OS.* Always present and mounted read-only.*/public static @NonNull File getRootDirectory() {return DIR_ANDR

JS和jQuery获取节点的兄弟,父级,子级元素

原文转自http://blog.csdn.net/duanshuyong/article/details/7562423 先说一下JS的获取方法,其要比JQUERY的方法麻烦很多,后面以JQUERY的方法作对比。 JS的方法会比JQUERY麻烦很多,主要则是因为FF浏览器,FF浏览器会把你的换行也当最DOM元素。 <div id="test"><div></div><div></div

vcpkg子包路径批量获取

获取vcpkg 子包的路径,并拼接为set(CMAKE_PREFIX_PATH “拼接路径” ) import osdef find_directories_with_subdirs(root_dir):# 构建根目录下的 "packages" 文件夹路径root_packages_dir = os.path.join(root_dir, "packages")# 如果 "packages"

Weex入门教程之4,获取当前全局环境变量和配置信息(屏幕高度、宽度等)

$getConfig() 获取当前全局环境变量和配置信息。 Returns: config (object): 配置对象;bundleUrl (string): bundle 的 url;debug (boolean): 是否是调试模式;env (object): 环境对象; weexVersion (string): Weex sdk 版本;appName (string): 应用名字;

MFC中App,Doc,MainFrame,View各指针的互相获取

纸上得来终觉浅,为了熟悉获取方法,我建了个SDI。 首先说明这四个类的执行顺序是App->Doc->Main->View 另外添加CDialog类获得各个指针的方法。 多文档的获取有点小区别,有时间也总结一下。 //  App void CSDIApp::OnApp() {      //  App      //  Doc     CDocument *pD

android两种日志获取log4j

android   log4j 加载日志使用方法; 先上图: 有两种方式: 1:直接使用架包 加载(两个都要使用); 架包:android-logging-log4j-1.0.3.jar 、log4j-1.2.15.jar  (说明:也可以使用架包:log4j-1.2.17.jar)  2:对架包输入日志的二次封装使用; 1:直接使用 log4j 日志框架获取日志信息: A:配置 日志 文

17 通过ref代替DOM用来获取元素和组件的引用

重点 ref :官网给出的解释是: ref: 用于注册对元素或子组件的引用。引用将在父组件的$refs 对象下注册。如果在普通DOM元素上使用,则引用将是该元素;如果在子组件上使用,则引用将是组件实例: <!-- vm.$refs.p will be the DOM node --><p ref="p">hello</p><!-- vm.$refs.child will be the c

react笔记 8-19 事件对象、获取dom元素、双向绑定

1、事件对象event 通过事件的event对象获取它的dom元素 run=(event)=>{event.target.style="background:yellowgreen" //event的父级为他本身event.target.getAttribute("aid") //这样便获取到了它的自定义属性aid}render() {return (<div><h2>{

react笔记 8-18 事件 方法 定义方法 获取/改变数据 传值

1、定义方法并绑定 class News extends React.Component {constructor(props) {super(props)this.state = {msg:'home组件'}}run(){alert("我是一个run") //方法写在类中}render() {return (<div><h2>{this.state.msg}</h2><button onCli

Linux下获取硬盘空间的大小

1. df 命令查看所有硬盘设备的信息 2. 查看指定路径的磁盘空间大小 代码获取空间大小案例: