借用了FlvParse这个工具和UI,结合官方文档分析出来的。
此工具挺好,就是不开源,而且对于Metadata Tag的数据也没有精确显示到每一个byte。实际使用当中还出现字段没有完全显示之类的。
其中遇到了和作者的一些分歧。所以写下我自己的总结,下面是这个工具的cnblog:
http://www.cnblogs.com/xxcainiao/archive/2010/01/12/1645408.html
===================================================================================
FLV是一个二进制文件,由文件头(FLV header)和很多tag组成。tag又可以分成三类:audio,video,script(metaData),分别代表音频流,视频流,脚本流(关键字或者文件信息之类)。
FLV Header
一般比较简单,包括文件类型之类的全局信息,如图:
文件类型 |
3bytes |
总是FLV(0x46 0x4C 0x56),否则... |
1、FLV 3bytes:0x46 0x4C 0x56标识FLV文件
2、版本号1bytes:0x01,表示FLV version 1
3、流信息 1bytes:0x04,表示成二进制位0000 0100,在这8个bit中,倒数第一位为1则表示有视频,倒数第三位为1表示有音频,这里表示有音频
4、Head的大小4bytes:0x00 0x00 0x00 0x09,整个文件头的长度,3 bytes + 1 bytes + 1 bytes + 4 bytes = 9
5、Head Tag长度 4bytes:0x00 0x00 0x00 0x00,一般为0,表示head标签
FLV Header之后便是FLV body,由很多tag组成的。
FLV文件里面帧的实体就是tag了。每个tag都可以分为两部分,第一部分包含是tag 类型信息,长度固定为15bytes
Metadata Tag
第一个tag为Metadata Tag[0x0000000D]
1、Tag Header 11bytes:
a、type(类型) 1bytes:此处为0x12即十进制的18
b、data size(数据区长度) 3bytes:此处为0x00 0x00 0xE0
c、time stamp(时间戳) 3bytes:此处为0x00 0x00 0xE0
d、expand time stamp(扩展时间戳) 1bytes:此处为0x00
e、stream id(流ID) 3bytes:此处为0x00 0x00 0xE0
2、Tag Data(大小为Tag Header中字段data size + 11[head的长度])
前18bytes定死,填0也行,不过一般保持默认
a、amf 1 type 1bytes:固定为0x02
b、string size 2bytes:固定为0x00 0x0A,即为10长度,表示字符串onMetaData的长度
c、string 10bytes(由上一个字段string size决定):固定为0x6F 0x6E 0x4D 0x65 0x74 0x64 0x44 0x61 0x74 0x61,表示字符串onMetaData
d、amf 2 type 1bytes:固定为0x08
e、array size 4bytes:表示接下来的metadata array data 中有多少组数据
综上:在metadata tag data部分的长度可以这么计算:
metadata array data size = data size(数据区长度) - 18bytes(上面几个字段的长度)
f、metadata array data:①前面2bytes表示,第N个数组的名字所占的bytes,继续读下去就可以知道数组的名字,例:
这里为0x00 0x08表示后面的8个bytes表示数组的名字
duration即为读出来的数组的名字信息
②跟着下去的1bytes表示这个数组的属性信息,
大体的类型介绍如下:
00类型-----随后的8bytes表示该属性对应的float值;
01类型-----随后的1bytes表示boolean,比如是否有视频为01表示“含有”的意思;
02类型-----后跟的2bytes表示字符串长度,然后再根据这个长度从后边的bytes中读取出字符串;
03类型-----数组信息的结束;
0A类型-----数据变量,占4bytes;
0B类型-----占10bytes,表示日期;
③metadata array data最后的3个bytes应为0x00 0x00 0x09表示metadata array data的结束
Pre Tag size:Metadata Tag总的长度 4bytes
可以通过以下公式来计算这个长度:
Pre Tag size = data size + 18 + 11;
事实证明:
把Metadata Tag data数据全部填上0也不影响数据的播放。但删掉后从新改写各个size却显示无法渲染= = 估计会和前面的flv header具有相关性。