威尼斯网址开户网站详解python进行mp3格式判断,详解pythonmp3格式

详解python进行mp3格式判断,详解pythonmp3格式

项目中使用mp3格式进行音效播放,遇到一个mp3文件在程序中死活播不出声音,最后发现它是wav格式的文件,却以mp3结尾。要对资源进行mp3格式判断,那么如何判断呢,用.mp3后缀肯定不靠谱,得从编码格式判断,方法如下:

1.mp3编码

MP3文件是一种流媒体文件格式,所以没有文件头。像AVI、WAV这种有文件头的格式,很好判断,他们都是RIFF开头的,只要进行RIFF字符串对比,就可以查出是否是AVI、WAV,而mp3就只能分析编码格式了。这里大概说mp3编码规则一下,详细的可用参考这篇文章

MP3 文件大体分为三部分:TAG_V2(ID3V2),音频数据,TAG_V1(ID3V1)

a). ID3V2
在文件开始的位置,以ID3开头,包含了作者,作曲,专辑等信息,长度不固定,扩展了ID3V1
的信息量,非必需

b).
一系列的音频数据的帧,在文件的中间位置,个数由文件大小和帧长决定;每个帧都以FFF开头,的长度可能不固定,也可能固定,由位率bitrate决定;每个帧又分为帧头和数据实体两部分;帧头记录了mp3
的位率,采样率,版本等信息,每个帧之间相互独立 。

c).
ID3V1在文件结尾的位置,以TAG开头,包含了作者,作曲,专辑等信息,长度为128Byte,非必须。

ID3V2

包含了作者,作曲,专辑等信息,长度不固定,扩展了ID3V1的信息量。

Frame

.

.

.

Frame

一系列的帧,个数由文件大小和帧长决定

每个FRAME的长度可能不固定,也可能固定,由位率bitrate决定

每个FRAME又分为帧头和数据实体两部分

帧头记录了mp3的位率,采样率,版本等信息,每个帧之间相互独立。

ID3V1

包含了作者,作曲,专辑等信息,长度为128BYTE。 

 也就是说,根据TAG_V2(ID3V2),音频数据,TAG_V1(ID3V1)三结构中的开头信息,便可以判断出是不是mp3编码的文件。

2.python代码

# coding: utf-8

'''
@author: BigFengFeng
@time: 16/12/21 下午6:10
@license: Apache Licence
@description:

'''

import os

#mp3filePath是否是mp3格式的
def isMp3Format(mp3filePath):
 #读取文件内字符串
 f = open(mp3filePath, "r");
 fileStr = f.read();
 f.close();
 head3Str = fileStr[:3];

 #判断开头是不是ID3
 if head3Str == "ID3":
  return True;

 #判断结尾有没有TAG
 last32Str = fileStr[-32:];
 if last32Str[:3] == "TAG":
  return True;

 #判断第一帧是不是FFF开头, 转成数字
 # fixme 应该循环遍历每个帧头,这样才能100%判断是不是mp3
 ascii = ord(fileStr[:1]);
 if ascii == 255:
  return True;

 return False;


#遍历folderPath看看是不是都是mp3格式的,
#是就true,不是就是false, 并返回是mp3的list,不是MP3的list
def isMp3FolderTraverse(folderPath):
 mp3List = [];
 notMp3List = [];
 isAllMpFormat = True;
 for dirpath, dirnames, filenames in os.walk(folderPath):
  for filename in filenames:
   path = dirpath + os.sep + filename;
   isMp3 = isMp3Format(path);
   #判断是不是mp3结尾的 并且 是mp3格式的
   if isMp3 == False and str.endswith(path, ".mp3") == True:
    # print("--warning: file " + path + " is not mp3 format!--");
    notMp3List.append(path);
    isAllMpFormat = False;
   else:
    mp3List.append(path);
 return isAllMpFormat, mp3List, notMp3List;


if __name__ == '__main__':
 isMp3Format("s_com_click1.mp3");
 isAllMp3, mp3List, notMp3List = isMp3FolderTraverse("sound");
 print isAllMp3;
 print mp3List;
 print notMp3List;

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持帮客之家。

项目中使用mp3格式进行音效播放,遇到一个mp3文件在程序中死活播不出声音,最后发现它是wa…

python进行mp3格式判断

   
 项目中使用mp3格式进行音效播放,遇到一个mp3文件在程序中死活播不出声音,最后发现它是wav格式的文件,却以mp3结尾。要对资源进行mp3格式判断,那么如何判断呢,用.mp3后缀肯定不靠谱,得从编码格式判断,方法如下:

python进行mp3格式判断,pythonmp3格式

项目中使用mp3格式进行音效播放,遇到一个mp3文件在程序中死活播不出声音,最后发现它是wav格式的文件,却以mp3结尾。要对资源进行mp3格式判断,那么如何判断呢,用.mp3后缀肯定不靠谱,得从编码格式判断,方法如下:

MP4(MPEG-4 Part 14)是一种常见的多媒体容器格式,它是在“ISO/IEC
14496-14”标准文件中定义的,属于MPEG-4的一部分,是“ISO/IEC
14496-12(MPEG-4 Part 12 ISO base media file
format)”标准中所定义的媒体格式的一种实现,后者定义了一种通用的媒体文件结构标准。MP4是一种描述较为全面的容器格式,被认为可以在其中嵌入任何形式的数据,各种编码的视频、音频等都不在话下,不过我们常见的大部分的MP4文件存放的AVC(H.264)MPEG-4(Part
2)
编码的视频和AAC编码的音频。MP4格式的官方文件后缀名是“.mp4”,还有其他的以mp4为基础进行的扩展或者是缩水版本的格式,包括:M4V,3GP,F4V等。

1.mp3编码

   
 MP3文件是一种流媒体文件格式,所以没有文件头。像AVI、WAV这种有文件头的格式,很好判断,他们都是RIFF开头的,只要进行RIFF字符串对比,就可以查出是否是AVI、WAV,而mp3就只能分析编码格式了。这里大概说mp3编码规则一下,详细的可用参考这篇文章

     MP3 文件大体分为三部分:TAG_V2(ID3V2),音频数据,TAG_V1(ID3V1)

     a). ID3V2 在文件开始的位置,以ID3开头,包含了作者,作曲,专辑等信息,长度不固定,扩展了ID3V1
的信息量,非必需 

     b).
一系列的音频数据的帧,在文件的中间位置,个数由文件大小和帧长决定;每个帧都以FFF开头,的长度可能不固定,也可能固定,由位率bitrate决定;每个帧又分为帧头和数据实体两部分;帧头记录了mp3
的位率,采样率,版本等信息,每个帧之间相互独立 。

     c). ID3V1在文件结尾的位置,以TAG开头,包含了作者,作曲,专辑等信息,长度为128Byte,非必须。

ID3V2

包含了作者,作曲,专辑等信息,长度不固定,扩展了ID3V1的信息量。

Frame

.

.

.

Frame

一系列的帧,个数由文件大小和帧长决定

每个FRAME的长度可能不固定,也可能固定,由位率bitrate决定

每个FRAME又分为帧头和数据实体两部分

帧头记录了mp3的位率,采样率,版本等信息,每个帧之间相互独立。

ID3V1

包含了作者,作曲,专辑等信息,长度为128BYTE。 

 也就是说,根据TAG_V2(ID3V2),音频数据,TAG_V1(ID3V1)三结构中的开头信息,便可以判断出是不是mp3编码的文件。

python进行mp3格式判断

   
 项目中使用mp3格式进行音效播放,遇到一个mp3文件在程序中死活播不出声音,最后发现它是wav格式的文件,却以mp3结尾。要对资源进行mp3格式判断,那么如何判断呢,用.mp3后缀肯定不靠谱,得从编码格式判断,方法如下:

1.mp3编码

首先看一下软件对于mp4文件的解析如图1所示:

2.python代码

 1 # coding: utf-8
 2 
 3 '''
 4 @author: BigFengFeng
 5 @time: 16/12/21 下午6:10
 6 @license: Apache Licence
 7 @description:
 8 
 9 '''
10 
11 import os
12 
13 #mp3filePath是否是mp3格式的
14 def isMp3Format(mp3filePath):
15     #读取文件内字符串
16     f =  open(mp3filePath, "r");
17     fileStr = f.read();
18     f.close();
19     head3Str = fileStr[:3];
20 
21     #判断开头是不是ID3
22     if head3Str == "ID3":
23         return True;
24 
25     #判断结尾有没有TAG
26     last32Str = fileStr[-32:];
27     if last32Str[:3] == "TAG":
28         return True;
29 
30     #判断第一帧是不是FFF开头, 转成数字
31     # fixme 应该循环遍历每个帧头,这样才能100%判断是不是mp3
32     ascii = ord(fileStr[:1]);
33     if ascii == 255:
34         return True;
35 
36     return False;
37 
38 
39 #遍历folderPath看看是不是都是mp3格式的,
40 #是就true,不是就是false, 并返回是mp3的list,不是MP3的list
41 def isMp3FolderTraverse(folderPath):
42     mp3List = [];
43     notMp3List = [];
44     isAllMpFormat = True;
45     for dirpath, dirnames, filenames in os.walk(folderPath):
46         for filename in filenames:
47             path = dirpath + os.sep + filename;
48             isMp3 = isMp3Format(path);
49             #判断是不是mp3结尾的 并且 是mp3格式的
50             if isMp3 == False and str.endswith(path, ".mp3") == True:
51                 # print("--warning: file " + path + " is not mp3 format!--");
52                 notMp3List.append(path);
53                 isAllMpFormat = False;
54             else:
55                 mp3List.append(path);
56     return isAllMpFormat, mp3List, notMp3List;
57 
58 
59 if __name__ == '__main__':
60     isMp3Format("s_com_click1.mp3");
62     isAllMp3, mp3List, notMp3List = isMp3FolderTraverse("sound");
63     print isAllMp3;
64     print mp3List;
65     print notMp3List;

 

1.mp3编码

   
 MP3文件是一种流媒体文件格式,所以没有文件头。像AVI、WAV这种有文件头的格式,很好判断,他们都是RIFF开头的,只要进行RIFF字符串对比,就可以查出是否是AVI、WAV,而mp3就只能分析编码格式了。这里大概说mp3编码规则一下,详细的可用参考这篇文章

     MP3
文件大体分为三部分:TAG_V2(ID3V2),音频数据,TAG_V1(ID3V1)

     a). ID3V2
在文件开始的位置,以ID3开头,包含了作者,作曲,专辑等信息,长度不固定,扩展了ID3V1
的信息量,非必需 

     b).
一系列的音频数据的帧,在文件的中间位置,个数由文件大小和帧长决定;每个帧都以FFF开头,的长度可能不固定,也可能固定,由位率bitrate决定;每个帧又分为帧头和数据实体两部分;帧头记录了mp3
的位率,采样率,版本等信息,每个帧之间相互独立 。

     c).
ID3V1在文件结尾的位置,以TAG开头,包含了作者,作曲,专辑等信息,长度为128Byte,非必须。

ID3V2

包含了作者,作曲,专辑等信息,长度不固定,扩展了ID3V1的信息量。

Frame

.

.

.

Frame

一系列的帧,个数由文件大小和帧长决定

每个FRAME的长度可能不固定,也可能固定,由位率bitrate决定

每个FRAME又分为帧头和数据实体两部分

帧头记录了mp3的位率,采样率,版本等信息,每个帧之间相互独立。

ID3V1

包含了作者,作曲,专辑等信息,长度为128BYTE。 

 也就是说,根据TAG_V2(ID3V2),音频数据,TAG_V1(ID3V1)三结构中的开头信息,便可以判断出是不是mp3编码的文件。

MP3文件是一种流媒体文件格式,所以没有文件头。像AVI、WAV这种有文件头的格式,很好判断,他们都是RIFF开头的,只要进行RIFF字符串对比,就可以查出是否是AVI、WAV,而mp3就只能分析编码格式了。这里大概说mp3编码规则一下,详细的可用参考这篇文章

威尼斯网址开户网站 1

2.python代码

 1 # coding: utf-8
 2 
 3 '''
 4 @author: BigFengFeng
 5 @time: 16/12/21 下午6:10
 6 @license: Apache Licence
 7 @description:
 8 
 9 '''
10 
11 import os
12 
13 #mp3filePath是否是mp3格式的
14 def isMp3Format(mp3filePath):
15     #读取文件内字符串
16     f =  open(mp3filePath, "r");
17     fileStr = f.read();
18     f.close();
19     head3Str = fileStr[:3];
20 
21     #判断开头是不是ID3
22     if head3Str == "ID3":
23         return True;
24 
25     #判断结尾有没有TAG
26     last32Str = fileStr[-32:];
27     if last32Str[:3] == "TAG":
28         return True;
29 
30     #判断第一帧是不是FFF开头, 转成数字
31     # fixme 应该循环遍历每个帧头,这样才能100%判断是不是mp3
32     ascii = ord(fileStr[:1]);
33     if ascii == 255:
34         return True;
35 
36     return False;
37 
38 
39 #遍历folderPath看看是不是都是mp3格式的,
40 #是就true,不是就是false, 并返回是mp3的list,不是MP3的list
41 def isMp3FolderTraverse(folderPath):
42     mp3List = [];
43     notMp3List = [];
44     isAllMpFormat = True;
45     for dirpath, dirnames, filenames in os.walk(folderPath):
46         for filename in filenames:
47             path = dirpath + os.sep + filename;
48             isMp3 = isMp3Format(path);
49             #判断是不是mp3结尾的 并且 是mp3格式的
50             if isMp3 == False and str.endswith(path, ".mp3") == True:
51                 # print("--warning: file " + path + " is not mp3 format!--");
52                 notMp3List.append(path);
53                 isAllMpFormat = False;
54             else:
55                 mp3List.append(path);
56     return isAllMpFormat, mp3List, notMp3List;
57 
58 
59 if __name__ == '__main__':
60     isMp3Format("s_com_click1.mp3");
62     isAllMp3, mp3List, notMp3List = isMp3FolderTraverse("sound");
63     print isAllMp3;
64     print mp3List;
65     print notMp3List;

 

python进行mp3格式判断
项目中使用mp3格式进行音效播放,遇到一个mp3文件在程序中死活播不出声音,最后…

MP3 文件大体分为三部分:TAG_V2(ID3V2),音频数据,TAG_V1(ID3V1)

图1

a). ID3V2
在文件开始的位置,以ID3开头,包含了作者,作曲,专辑等信息,长度不固定,扩展了ID3V1
的信息量,非必需

从图中可以看出这个视频文件第一层有4部分,每一部分都是一个box,分别为:ftype,moov,free,mdat。其实mp4文件是有许多的box组成的。如图2所示:

b).
一系列的音频数据的帧,在文件的中间位置,个数由文件大小和帧长决定;每个帧都以FFF开头,的长度可能不固定,也可能固定,由位率bitrate决定;每个帧又分为帧头和数据实体两部分;帧头记录了mp3
的位率,采样率,版本等信息,每个帧之间相互独立 。

威尼斯网址开户网站 2

c).
ID3V1在文件结尾的位置,以TAG开头,包含了作者,作曲,专辑等信息,长度为128Byte,非必须。

图2

ID3V2

包含了作者,作曲,专辑等信息,长度不固定,扩展了ID3V1的信息量。

Frame

.

.

.

Frame

一系列的帧,个数由文件大小和帧长决定

每个FRAME的长度可能不固定,也可能固定,由位率bitrate决定

每个FRAME又分为帧头和数据实体两部分

帧头记录了mp3的位率,采样率,版本等信息,每个帧之间相互独立。

ID3V1

包含了作者,作曲,专辑等信息,长度为128BYTE。 

box的基本结构如图3所示,其中,size指明了整个box所占用的大小,包括header部分,type指明了box的类型。如果box很大(例如存放具体视频数据的mdat
box),超过了uint32的最大数值,size就被设置为1,并用接下来的8位uint64来存放大小。

 也就是说,根据TAG_V2(ID3V2),音频数据,TAG_V1(ID3V1)三结构中的开头信息,便可以判断出是不是mp3编码的文件。

威尼斯网址开户网站 3

2.python代码

图3

# coding: utf-8
 
”’
@author: BigFengFeng
@time: 16/12/21 下午6:10
@license: Apache Licence
@description:
 
”’
 
importos
 
#mp3filePath是否是mp3格式的
defisMp3Format(mp3filePath):
 #读取文件内字符串
 f=open(mp3filePath,"r");
 fileStr=f.read();
 f.close();
 head3Str=fileStr[:3];
 
 #判断开头是不是ID3
 ifhead3Str=="ID3":
  returnTrue;
 
 #判断结尾有没有TAG
 last32Str=fileStr[-32:];
 iflast32Str[:3]=="TAG":
  returnTrue;
 
 #判断第一帧是不是FFF开头, 转成数字
 # fixme 应该循环遍历每个帧头,这样才能100%判断是不是mp3
 ascii=ord(fileStr[:1]);
 ifascii==255:
  returnTrue;
 
 returnFalse;
 
 
#遍历folderPath看看是不是都是mp3格式的,
#是就true,不是就是false, 并返回是mp3的list,不是MP3的list
defisMp3FolderTraverse(folderPath):
 mp3List=[];
 notMp3List=[];
 isAllMpFormat=True;
 fordirpath, dirnames, filenamesinos.walk(folderPath):
  forfilenameinfilenames:
   path=dirpath+os.sep+filename;
   isMp3=isMp3Format(path);
   #判断是不是mp3结尾的 并且 是mp3格式的
   ifisMp3==Falseandstr.endswith(path,".mp3")==True:
    # print("–warning: file " + path + " is not mp3 format!–");
    notMp3List.append(path);
    isAllMpFormat=False;
   else:
    mp3List.append(path);
 returnisAllMpFormat, mp3List, notMp3List;
 
 
if__name__==’__main__’:
 isMp3Format("s_com_click1.mp3");
 isAllMp3, mp3List, notMp3List=isMp3FolderTraverse("sound");
 printisAllMp3;
 printmp3List;
 printnotMp3List;

一个mp4文件有可能包含非常多的box,在很大程度上增加了解析的复杂性,这个网页上http://mp4ra.org/atoms.html记录了一些当前注册过的box类型。看到这么多box,如果要全部支持,一个个解析,怕是头都要爆了。还好,大部分mp4文件没有那么多的box类型,下图就是一个简化了的,常见的mp4文件结构如图4所示

威尼斯网址开户网站 4

图4

一般来说,解析媒体文件,最关心的部分是视频文件的宽高、时长、码率、编码格式、帧列表、关键帧列表,以及所对应的时戳和在文件中的位置,这些信息,在mp4中,是以特定的算法分开存放在stbl
box下属的几个box中的,需要解析stbl下面所有的box,来还原媒体信息。下表是对于以上几个重要的box存放信息的说明:

威尼斯网址开户网站 5

威尼斯网址开户网站,图5

其实除了moov在前面的还有moov在后面的情况,如果要实现边下边播,就得把moov的box移到前面,因为只有获得了moov的信息,播放器才可以获取到播放器的信息来进行播放。具体代码如下:

– (NSData*)exchangestco:(NSMutableData*) moovdata{

inti, atom_size, offset_count, current_offset;

NSString*atom_type;

longlongmoov_atom_size = moovdata.length;

Byte*buffer = (Byte*)malloc(5);

buffer[4] =0;

Byte*buffer01 = (Byte*)malloc(moov_atom_size);

[moovdatagetBytes:buffer01 length:moov_atom_size];

for(i =4; i < moov_atom_size -4; i++) {

NSRangerange;

range.location= i;

range.length=4;

[moovdatagetBytes:buffer range:range];

atom_type = [selftosType:buffer];

if([atom_typeisEqualToString:@”stco”]) {

range.location= i-4;

range.length =4;

[moovdatagetBytes:bufferrange:range];

atom_size = [selftoSize:buffer];

if(i + atom_size -4> moov_atom_size) {

WBLog(LOG_ERROR,@”error i + atom_size – 4 > moov_atom_size”);

returnnil;

}

range.location= i+8;

range.length=4;

[moovdatagetBytes:bufferrange:range];

offset_count = [selftoSize:buffer];

for(intj =0; j < offset_count; j++) {

range.location= i +12+ j *4;

range.length=4;

[moovdatagetBytes:bufferrange:range];

current_offset= [selftoSize:buffer];

current_offset += moov_atom_size;

buffer01[i +12+ j *4+0] = (Byte) ((current_offset >>24)
&0xFF);

buffer01[i +12+ j *4+1] = (Byte) ((current_offset >>16)
&0xFF);

buffer01[i +12+ j *4+2] = (Byte) ((current_offset >>8) &0xFF);

buffer01[i +12+ j *4+3] = (Byte) ((current_offset >>0) &0xFF);

}

i += atom_size -4;

}

elseif([atom_typeisEqualToString:@”co64″]) {

range.location= i-4;

range.length=4;

[moovdatagetBytes:bufferrange:range];

atom_size = [selftoSize:buffer];

if(i + atom_size -4> moov_atom_size) {

WBLog(LOG_ERROR,@”error i + atom_size – 4 > moov_atom_size”);

returnnil;

}

range.location= i+8;

range.length=4;

[moovdatagetBytes:bufferrange:range];

offset_count = [selftoSize:buffer];

for(intj =0; j < offset_count; j++) {

range.location= i +12+ j *8;

range.length=4;

[moovdatagetBytes:bufferrange:range];

current_offset = [selftoSize:buffer];

current_offset += moov_atom_size;

buffer01[i +12+ j *8+0] = (Byte)((current_offset >>56) &0xFF);

buffer01[i +12+ j *8+1] = (Byte)((current_offset >>48) &0xFF);

buffer01[i +12+ j *8+2] = (Byte)((current_offset >>40) &0xFF);

buffer01[i +12+ j *8+3] = (Byte)((current_offset >>32) &0xFF);

buffer01[i +12+ j *8+4] = (Byte)((current_offset >>24) &0xFF);

buffer01[i +12+ j *8+5] = (Byte)((current_offset >>16) &0xFF);

buffer01[i +12+ j *8+6] = (Byte)((current_offset >>8) &0xFF);

buffer01[i +12+ j *8+7] = (Byte)((current_offset >>0) &0xFF);

}

i += atom_size -4;

}

}

NSData*moov = [NSDatadataWithBytes:buffer01length:moov_atom_size];

free(buffer);

free(buffer01);

returnmoov;

}

相关文章