Python实现FLV视频拼接功能
文章摘要
本文简单说明了FLV文件的格式,以此为出发点,使用Python实现FLV视频的拼接。
一.FLV文件格式
关于FLV文件格式的解析网上有诸多文章,在这里就简单介绍一下需要了解的部分,以便读者更好地明白各段代码的功能。
FLV文件是由文件头(Header)和文件体(Body)按顺序拼接而成。审查FLV内容时,以二进制方式读取内容。
Header:文件头表明了文件的封装格式为FLV,存储对象为音频、视频或两者。
以下为FLV文件的Header,共9个字节:
b'FLV\x01\x05\x00\x00\x00\t'
前3个字节(FLV)说明这是一个FLV文件
第4个字节(\x01)为版本号,固定为1
第5个字节(\x05)表明存储对象,需将其转化成二进制(00000101)查看,左、右边的1分别表示文件含有音频和视频
后4个字节(\x00\x00\x00\t)表示文件头的长度,其值固定为9
Body:文件体由若干个Tag组成,除了第一个,每个Tag是由头部(11字节)、主体(不定长)和尾部(4字节)组成。第一个Tag只有尾部。
Tag又分为3类,脚本(scripts)、音频(audio)和视频(video)。通常第2个Tag为脚本类型,且只有一个,后续的都是音视频类型。
以下为脚本Tag的部分,作为示例介绍一下:
头部:b'\x12\x00\tb\x00\x00\x00\x00\x00\x00\x00'
第1个字节(\x12)表示Tag类型,脚本类型的对应值为18,音频为8,视频为9
第2-4个字节(\x00\tb)表示Tag主体的长度,此处为2402
第5-7个字节(\x00\x00\x00)为时间戳,脚本类型的时间戳通常为0
第8个字节(\x00)是时间戳的扩展,当前3个字节不够用时会用这个字节当作大端
后3个字节(\x00\x00\x00)是Streamid,固定为0
主体:脚本Tag的主体包含FLV视频的基本信息,如时长、大小、分辨率等,比较复杂,在此不作介绍
尾部:b'\x00\x00\tm'
固定4字节,表示Tag头部加主体的长度,即11+2402=2413
二.FLV视频拼接
将多个FLV视频合成一个可以正常播放的视频,便足够满足大部分的需求。因此,在接下来的拼接过程中,不会对FLV进行细致入微的调整,达到基本要求即可。
设置阅读器
阅读器可以使我们很方便地读取文件内容。
classReader(): def__init__(self,content):#content(bytes):FLV文件的二进制内容 self.content=content self.start=0 self.eof=False#判断是否已读完全部内容 self.length=len(self.content) defread(self,n=1): #设置if语句防止过度读取内容 ifself.length>(self.start+n): out=self.content[self.start:self.start+n] self.start+=n else: out=self.content[self.start:] self.eof=True returnout
向新建FLV文件写入Header和Tag
在这里假设要拼接的视频基本信息相似,即都含有音视频,分辨率、码率等相同或相近。
为了生成一个可以正常播放的FLV视频,Header和Tag是必不可少的。我们可以选取第一个FLV的文件头写入新建FLV中,然后依次将修改过时间戳的Tag写入其中,便可达到拼接目的。
defadd_flv(flv,target,videoTimeStamp,audioTimeStamp):#修改并添加Tag的函数 withopen(flv,'rb')asf: content=f.read() reader=Reader(content) header=reader.read(13) withopen(target,'ab')asf: whilenotreader.eof:#一直读取直到读完,此时reader.eof=True dataType=reader.read(1) dataSize=reader.read(3) timeStamp=int.from_bytes(reader.read(3),'big')#将3字节转换成整数 headerRemained=reader.read(4) ifdataType==b'\t':#视频 timeStamp+=videoTimeStamp videoTS=timeStamp ifdataType==b'\x08':#音频 timeStamp+=audioTimeStamp audioTS=timeStamp timeStamp=timeStamp.to_bytes(3,'big')#将整数转换成3字节 tagHeader=dataType+dataSize+timeStamp+headerRemained tagData_andSize=reader.read(int.from_bytes(dataSize,'big')+4) f.write(tagHeader) f.write(tagData_andSize) returnvideoTS,audioTS defmerge_flv(flvs,target):#主函数 videoTS=0 audioTS=0 fori,flvinenumerate(flvs): withopen(flv,'rb')asf: content=f.read() reader=Reader(content) header=reader.read(13)#flvHeader+tagSize0 ifi==0:#写入第1个FLV视频的文件头 withopen(target,'wb')asf: f.write(header) videoTS,audioTS=add_flv(flv,target,videoTS,audioTS)
拼接
importtime since=time.time() flvs=['m1.flv','m2.flv','m3.flv','m4.flv']#视频大小:45MB,20MB,59MB,54MB target='t.flv' merge_flv(flvs,target) end=time.time() print('Mergingflvstakes{:.2f}s'.format(end-since)) #Mergingflvstakes0.88s
可以看到,拼接4个共178MB视频用时0.88秒。
总结
FLV文件格式还是比较简明的,对数据的要求也是比较宽松的,即便没有对Scripts里的参数作调整,拼接后的视频依然能够正常播放。
不过,拼接的视频是有不少隐形问题,如到视频末尾可能会出现音画不同步(0.5秒左右)的现象,以及不能够方便地分离出完整的视频和音频。
以上所述是小编给大家介绍的Python实现FLV视频拼接功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对毛票票网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。