RTMP协议与 FLV 又有什么关系呢?实际 上,FLV文件与 RTMP 之间是“近亲”关系,甚至比“近亲”还要近,亲得就好像是“一个人”似的。
FLV文件格式
我们先来看一下 FLV 的文件格式,如下图所示:
阅读FLV 格式规范文档 从图中我们可以看出,FLV 文件格式由 FLV Header 和 FLV Body 两部分组成。其中,FLV Header 由 9 个字节组成,Body由 Pre TagSize 和 Tag 组 成。
为使你对它们有一个更清楚的认知,下面我们就来详细介绍一下 FLV Header 和 FLV Body。
1. FLV Header
它由 9 个字节组成:3个字节的 “F”“L”“V”字母,用于标记该文件是 FLV 文件;1个字节的 Version, 指明使用的 FLV 文件格式的版本;1个字节的 Type 标识,用于表明该FLV文件中是否包括音频或视频; 4个 字节的 FLV Header ⻓度,由于 FLV 文件格式头是固定 9 个字节,这个字段设置得有点多余。
Type 标识(TypeFlag)又可以细分为: 1bit 用于标识FLV文件中是否有音频数据;1bit标识FLV文件中是否有视频数据;如果两个 bit 位同时置 1,说明该FLV 文件中既有音频数据又有视频数据,这也是通常情况下 FLV Header 的设置;除了两个 bit 的音视频数据标识外,其他位都是预留位,必须全部置0。详细的含义可 以参考下面张图表:
这张图表清晰地表达了 FLV Header 中每个域所占的字节以及该域的具体含义。
2. FLV Body
从“FLV文件格式结构图”我们可以看出,FLV Body 是由多个 Previous TagSize 和 Tag 组成的。其含义如 下图表所示,其中PreviousTagSize 占 4个字节,表示前一个 Tag 的大小。这里需要注意的是,第一个 Previous TagSize 比较特殊,由于它前面没有 Tag 数据,所以它的值必须为 0。
接下来我们再来看一下 FLV 中的 Tag,从 FLV文件格式结构图中我们可以看到 Tag 由两部分组成,即 Tag Header 和 Tag Data 。
Tag Header 各字段的含义如下图所示:
- TagType,占1个字节,表示该Tag的类型,可以是音频、视频和脚本。如果类型为音频,说明这个Tag存 放的是音频数据;如果类型是视频,说明存放的是视频数据。
- DataSize,占3个字节,表示音频/视频数据的⻓度。
- Timestamp和扩展Timestamp,一共占4个字节,表示数据生成时的时间戳。
- StreamID,占3个字节,总是为 0。
而Tag Data 中存放的数据,是根据 TagType 中的类型不同而有所区别的。也就是说,假如 TagType 指定的 是音频,那么 Tag Data中存放的就是音频数据;如果 TagType 指定的是视频,则Tag Data中存放的就是视 频数据。
无论TagData中存放的是音频数据还是视频数据,它们都是由 Header 和 Data 组成。也就是说,如果该 Tag 是一个音频 Tag ,那么它的数据就是由“AudioHeader + AudioData”组成;如果是一个视频 Tag,则它的数据是由“VideoHeader + VideoData”组成。
翻看RTMP协议,查看它的 6.1.1 小节,你会发现它定义的 RTMP Message Header 与 Tag Header 是一模一样的。下图是我从 RTMP 协议中截取的协议头:
因此,我们可以说FLV文件就是由“FLV Header + RTMP数据”构成的。这也揭开了 FLV与 RTMP之间的关 系秘密,即 FLV 是在 RTMP 数据之上加了一层“⻢甲”。