关于mp3文件结构解析的问题

最近在用qt5编写播放器时遇到个问题,在Windows端需要解析音频文件音谱来绘制曲线图,但发现Windows端的库要么太老、要么太弱鸡、要么得自己编译,无奈花了一天时间在下载、编译taglib、学习ffmpeg路上,到最后都是以动态库调用时出现未知函数告终(mingw_gcc),原来qt本身就有对这些的封装,可采用qmediaplaer代替他们,只不过需要安装解码器才行,但qmediaplaer并不支持ffmpeg、taglib的音频文件直接操作功能,这不得已自己重写一版。。。

参考文章
MP3文件结构解析(超详细)
要一个解析MP3的代码要C语言的 要自己写的不要网上的
Inside the MP3 Codec - Page 11
lets-build-mp3-decoder
Reading MP3 files [closed]

C语言结构体为

ID3V2

//起始位置:0x0-0x9
struct ID3V2Header{
    
    
	char Header[3];    /*必须为“ID3”否则认为标签不存在*/
	char Ver[1];         /*版本号ID3V2.3 就记录3*/
	char Revision[1];     /*副版本号此版本记录为0*/
	char Flag[1];        /*标志字节,只使用高三位,其它位为0 */
	char Size[4];      /*标签大小*/
};
//起始位置:0x10-0x19
struct ID3V2INFO{
    
    
	char ID[4]; /*标识帧,说明其内容,例如作者/标题等*/
	char Size[4]; /*帧内容的大小,不包括帧头,不得小于1*/
	char Flags[2]; /*标志帧,只定义了6 位*/
};


ID3V1

//起始位置:-128
struct ID3V1HEAD{
    
    
	char tag[3];//tag占用3字节
	char musicname[30];//音乐标签占30字节
	char artist[30];//歌曲作者占30字节
	char album[30];//发行专辑占30字节
	char year[4];//发行年份占4字节
	char generic[30];//一些音乐的介绍以及其他杂七杂八内容占30字节
	char type[1];//类型,占1字节
};

part1:ID3v1版本可以通过音乐文件末尾倒数第128字节开始进行数据截取,这也是最简单的

//用于解析mp3文件,采用id3v1取倒数128固定位置数据
MP3INFO MediaTools::toMp3Info(QString path) {
    
    
  int part = 128;
  QString tag, musicname, artist, album, year, generic, musictype;
  QFile f(path);
  f.open(QFile::ReadOnly);//以只读方式打开文件
  f.seek(f.size() - part);//从倒数128开始读取数据
  part = part - 3;
  tag = QString::fromLocal8Bit(f.readLine(4));//前3字节为tag标签,这里为什么用4,因为这是qt
  f.seek(f.size() - part);
  part = part - 30;//这里减去30,是为了给下面往后移动30字节读取内容
  musicname = QString::fromLocal8Bit(f.readLine(31));
  f.seek(f.size() - part);
  part = part - 30;
  artist = QString::fromLocal8Bit(f.readLine(31));
  f.seek(f.size() - part);
  part = part - 30;
  album = QString::fromLocal8Bit(f.readLine(31));
  f.seek(f.size() - part);
  part = part - 4;
  year = QString::fromLocal8Bit(f.readLine(5));
  f.seek(f.size() - part);
  part = part - 30;
  generic = QString::fromLocal8Bit(f.readLine());
  f.seek(f.size() - 1);
  musictype = QString::fromLocal8Bit(f.readLine(1));
  f.close();
  //返回mp3info对象,可以用结构体代替
  return MP3INFO(tag, musicname, artist, album, year, generic, musictype);
}

part2:ID3v2.3版本在国内只有原理讲解文章与一些开源第三方库可供参考
我这里实现了一版非常烂的demo,望各位懂这方面的大佬帮忙指点指点

void MediaTools::seekFiles(int &len, QFile &f) {
    
    
  QByteArray by;//创建一个字节数组对象
  f.seek(len);//设置文件读取起始位置
  //这个为音乐文件里0x21起始头部的帧信息头部,占4字节
  qDebug() << "seekFiles  0 ::" << QString::fromLocal8Bit(f.readLine(5));
  f.seek(len + 4);
  //这个为音乐文件里帧内容大小
  by = f.read(4);
  qDebug() << "seekFiles  1 ::" << by;
  f.seek(len + 4 + 4);
  qDebug() << "seekFiles  2 ::" << QString::fromLocal8Bit(f.readLine(3));
  int FSize;
  //帧内容大小计算公式
  FSize = by[0] * 0x100000000 + by[1] * 0x10000 + by[2] * 0x100 + by[3];
  qDebug() << "seekFiles  fsize ::" << FSize;
  f.seek(len + 11);
  qDebug() << "seekFiles  data ::" << QString::fromLocal8Bit(f.readLine());
  len = len + 10 + FSize;
  qDebug() << "next positi ::: " << len << "\n";
}

//第一次调用位置
void MediaTools::Mp3ID3V2_3(QString path) {
    
    
  QFile f(path);
  f.open(QFile::ReadOnly);
  QByteArray by;
  //读取0x0-0x9位置的数据
  qDebug() << "0 ::" << QString::fromLocal8Bit(f.readLine(4));//tag标识
  f.seek(3);
  qDebug() << "1 ::" << QString::fromLocal8Bit(f.readLine(2));
  f.seek(4);
  qDebug() << "2 ::" << QString::fromLocal8Bit(f.readLine(2));
  f.seek(5);
  qDebug() << "3 ::" << QString::fromLocal8Bit(f.readLine(2));
  f.seek(6);
  //帧数据内容大小,这里的大小为所有帧的内容大小,也就是说,在这个地址往后偏移1位就是音频数据了
  by = f.read(4);
  qDebug() << "4 ::" << by;
  int len = 10;
  int total_size;

  total_size = (by[0] & 0x7F) * 0x200000 + (by[1] & 0x7F) * 0x400 +
               (by[2] & 0x7F) * 0x80 + (by[3] & 0x7F);
  qDebug() << total_size;
  //通过不断的调用来获取音频文件里TIT2、TPE1、TALB等内容信息
  seekFiles(len, f);
  seekFiles(len, f);
  seekFiles(len, f);
  seekFiles(len, f);
  seekFiles(len, f);
  seekFiles(len, f);
  seekFiles(len, f);
  seekFiles(len, f);

  f.close();
}

猜你喜欢

转载自blog.csdn.net/sorry_my_life/article/details/106647967
今日推荐