WAVE文件格式说明表:
| 偏移地址 | 字节数 | 数据类型 | 内 容 |
文件头 | 00H | 4 | char | "RIFF"标志 |
04H | 4 | long int | 文件长度 | |
08H | 4 | char | "WAVE"标志 | |
0CH | 4 | char | "fmt"标志 | |
10H | 4 |
| 过渡字节(不定) | |
14H | 2 | int | 格式类别(10H为PCM形式的声音数据) | |
16H | 2 | int | 通道数,单声道为1,双声道为2 | |
18H | 4 | int | 采样率(每秒样本数),表示每个通道的播放速度, | |
1CH | 4 | long int | 波形音频数据传送速率,其值为通道数×每秒数据位数×每样本的数据位数/8。播放软件利用此值可以估计缓冲区的大小。 | |
20H | 2 | int | 数据块的调整数(按字节算的),其值为通道数×每样本的数据位值/8。播放软件需要一次处理多个该值大小的字节数据,以便将其值用于缓冲区的调整。 | |
22H | 2 |
| 每样本的数据位数,表示每个声道中各个样本的数据位数。如果有多个声道,对每个声道而言,样本大小都一样。 | |
24H | 4 | char | 数据标记符"data" | |
28H | 4 | long int | 语音数据的长度 |
PCM数据的存放方式:
| 样本1 | 样本2 | ||
8位单声道 | 0声道 | 0声道 | ||
8位立体声 | 0声道(左) | 1声道(右) | 0声道(左) | 1声道(右) |
16位单声道 | 0声道低字节 | 0声道高字节 | 0声道低字节 | 0声道高字节 |
16位立体声 | 0声道(左)低字节 | 0声道(左)高字节 | 1声道(右)低字节 | 1声道(右)高字节 |
读取wav文件所有字段,并原封不动写入一个新的wav文件: wav_write.cpp
#include#include #include #include #include #include #include #include using namespace std;struct my_wave{ unsigned long chunckId; unsigned long chunkSize; unsigned long format; unsigned long subChunk1Id; unsigned long subChunk1Size; unsigned short audioFormat; unsigned short numChannels; unsigned long sampleRate; unsigned long byteRate; unsigned short blockAlign; unsigned short bitsPerSampel; unsigned long subChunk2Id; unsigned long subChunk2Size; unsigned char *z_data;};int main(int argc, char **argv){ ifstream fs; ofstream ofs; my_wave MW; //读取原始文件 fs.open("F:\\wave.wav", ios::binary | ios::in); //这句需要修改成您的文件路径 ofs.open("F:\\waveout.wav", ios::binary | ios::out); //读取的文件写入新文件 //读取wav文件的所有字段 fs.seekg(0); fs.read((char*)&MW.chunckId, sizeof(MW.chunckId)); fs.seekg(0x04); fs.read((char*)&MW.chunkSize, sizeof(MW.chunkSize)); fs.seekg(0x08); fs.read((char*)&MW.format, sizeof(MW.format)); fs.seekg(0x0c); fs.read((char*)&MW.subChunk1Id, sizeof(MW.subChunk1Id)); fs.seekg(0x10); fs.read((char*)&MW.subChunk1Size, sizeof(MW.subChunk1Size)); fs.seekg(0x14); fs.read((char*)&MW.audioFormat, sizeof(MW.audioFormat)); fs.seekg(0x16); fs.read((char*)&MW.numChannels, sizeof(MW.numChannels)); fs.seekg(0x18); fs.read((char*)&MW.sampleRate, sizeof(MW.sampleRate)); fs.seekg(0x1c); fs.read((char*)&MW.byteRate, sizeof(MW.byteRate)); fs.seekg(0x20); fs.read((char*)&MW.blockAlign, sizeof(MW.blockAlign)); fs.seekg(0x22); fs.read((char*)&MW.bitsPerSampel, sizeof(MW.bitsPerSampel)); fs.seekg(0x24); fs.read((char*)&MW.subChunk2Id, sizeof(MW.subChunk2Id)); fs.seekg(0x28); fs.read((char*)&MW.subChunk2Size, sizeof(MW.subChunk2Size)); MW.z_data = new unsigned char[MW.subChunk2Size]; fs.seekg(0x2c); fs.read((char *)MW.z_data, sizeof(char)*MW.subChunk2Size); //输出wav文件的所有字段,以供测试 cout << MW.chunckId << endl; cout << MW.chunkSize << endl; cout << MW.format << endl; cout << MW.subChunk1Id << endl; cout << MW.subChunk1Size << endl; cout << MW.audioFormat << endl; cout << MW.numChannels << endl; cout << MW.sampleRate << endl; cout << MW.byteRate << endl; cout << MW.blockAlign << endl; cout << MW.bitsPerSampel << endl; cout << MW.subChunk2Id << endl; cout << MW.subChunk2Size << endl; cout << "采样数据:" << endl; for (int k = 0; k < MW.subChunk2Size; k++){ printf("%x ",MW.z_data[k]); } //将每个字段原封不动得写入新的.wav文件中 ofs.write((char*)&MW.chunckId, sizeof(MW.chunckId)); ofs.write((char*)&MW.chunkSize, sizeof(MW.chunkSize)); ofs.write((char*)&MW.format, sizeof(MW.format)); ofs.write((char*)&MW.subChunk1Id, sizeof(MW.subChunk1Id)); ofs.write((char*)&MW.subChunk1Size, sizeof(MW.subChunk1Size)); ofs.write((char*)&MW.audioFormat, sizeof(MW.audioFormat)); ofs.write((char*)&MW.numChannels, sizeof(MW.numChannels)); ofs.write((char*)&MW.sampleRate, sizeof(MW.sampleRate)); ofs.write((char*)&MW.byteRate, sizeof(MW.byteRate)); ofs.write((char*)&MW.blockAlign, sizeof(MW.blockAlign)); ofs.write((char*)&MW.bitsPerSampel, sizeof(MW.bitsPerSampel)); ofs.write((char*)&MW.subChunk2Id, sizeof(MW.subChunk2Id)); ofs.write((char*)&MW.subChunk2Size, sizeof(MW.subChunk2Size)); ofs.write((char*)MW.z_data, sizeof(char)*MW.subChunk2Size); fs.close(); ofs.close(); delete[] MW.z_data; system("pause");}