转自: http://blog.youkuaiyun.com/felix__/article/details/4121644
介绍下flv的格式:
flv文件是个二进制文件,由一个文件头(flv header)和许多个tag块(统一叫flv body)组成。
*********
1.文件头(flv header),通常是9个字节,主要就是用于表示文件是flv格式,没有什么有价值的信息。具体格式:
格式名称 长度 说明
文件类型 3字节 就是一个ascll编码的"FLV"字符串。
版本 1字节 一般是0x01,表示flv version是 1
流信息 1字节 倒数第一位是1表示有视频,倒数第三位是1表示有音频。其余位通常是0
header长度 4字节 保存整个文件头的长度。一般是9。
扩展信息 不确定 一般都没有扩展信息;如果有,其长度可以从"header长度"里计算出来
了。实际看了好几个视频,都没有这一信息。
*********
2.flv body。也就是flv的主体,这只是个概念,用来统称除掉flv header之外的部分,即一系列的tag。每个tag的格式都是固定,具体参考下面。
*********
3.tag。flv里面帧的实体就是tag了。每个tag都可以分为两部分,第一部分包含是tag 类型信息,长度固定为15字节;第二部分为tag data,也就是flv的数据(有音频,视频,脚本等三类数据),其长度由第一部分的特定位给出。具体格式及其含义如下:
上一个tag的长度 4个字节 这个长度包括了tag的两个部分,对于第一帧这个值为0。主要用于从文件尾逆向处理每帧。所以在最后一帧(tag)的后面会有一个previoustagsize(上一帧长度),才是文件结束,要注意这点。
tag种类 1字节 标志这个tag的种类。8是音频,9是视频,18是脚本。目前只有这三类tag。
data域长度 3字节 指出tag data(上面提到的第二部分)的总长度。
时间戳 3字节 此tag的时间戳。整数,单位是毫秒。对于脚本型的tag,这个字段总是0。
时间戳扩展位 1字节 用于把时间戳扩展为4字节(3+1),目前看来用不了这么长的时间戳,我估算了一下,3 字节能表示大约4.6个小时的长度,4字节能表示74小时,普通flv文件的时间长度都不太可能超过4.6个小时了,所以这个字段一般也是全0了
流ID 3字节 全0。英文是StreamID,没有找到更多的说明信息了,可以认为是保留位吧。
Data 由上面指出 flv的数据区。长度在"data域长度"给出。
到这里,flv的格式没有具体说明的就是tag中的data域,不过这个域的长度我们已经可以获得。切割时先拷贝flv header,然后一个一个地拷贝tag。计算好已经拷贝的总大小(或是时间长度),很容易就能按大小(或是时间长度)切割出文件来。当然,tag data域里有些信息我们可能是感兴趣的,比如一个视频的tag是不是关键帧。这就需要区解读data域。切割的时候,还要注意修改每帧的时间戳(每个文件的第一帧都应该从0开始)。flv文件的第一帧往往是被称为metadata 的脚本tag,在这个tag里有整个flv的时长(由一个duration的字符串跟着64位的dobule值,单位是秒),播放器会去读取这个数据,所以切割的时候也要保证这个数据的正确性。flv里的数据多数是按网络序存储在文件中的,读到本机处理前要逆序,处理后也要逆序再存回去。不做详细分析了,可以参考官方的文档《video_file_format_spec_v9》。 细心决定成败。
参考文献:(网页有可能过期)
http://noahgenius.javaeye.com/blog/164782
官方文档《video_file_format_spec_v9》