有了WAV头文件的格式,要生成一个WAV格式的声音文件就比较简单了,只要将头文件输出,在简单生成一点内容就可以了。为了简单,就生成一个单声道,11025HZ的,长度一秒的声音文件。

那么开做,要生成一个文件,那么文件的输出流<fstream>肯定要用到,参考了一下资料,简单整了如下的一个程序:

#include <iostream>
#include<fstream> //文件输出
#include<math.h> //主要生产波形使用SIN函数,所以弄了这个头

using namespace std;
float dou=256; //声音dou的频率264,这里为了凑整数用的256 HZ  

struct WavHead{
 char RIFF[4];    //头部分那个RIFF
 long int size0;//存的是后面所有文件大小
 char WAVE[4];
 char FMT[4];
 long int size1;//存的是fmt保存的大小,包含这之后,data前面几个,共16个
 short int fmttag;
 short int channel;
 long int samplespersec;//每秒采样数,用的是11025
 long int bytepersec;
 short int blockalign;
 short int bitpersamples;
 char DATA[4];
 long int size2;//剩下文件大小,也就是声音采样是大小,因为是一秒钟的内容,那么就是11025了。

};

int main() {

    WavHead head={{'R','I','F','F'},0,{'W','A','V','E'},{'f','m','t',' '},16,
            1,1,11025,11025,1,8,{'d','a','t','a'},
            0};//初始化,没将声音采样文件大小赋值,后面修改
    head.size0=11025+16+8;
    head.size2=11025;
    char body[head.size2];//打算将采样数据放在这个里面
    int i,i2;//为了声音不是很单调,用了两个循环,里面一层i是生成声音函数,外面那层i2,是生成不同频率的。
    float a=(head.samplespersec/dou);//是dou持续时间(1/dou)除以采样数(1/head.samplespersec),也就是一个dou的波形里有a个采样点,这里是43。
    for (i2=1;i2<=2;i2++){//这里弄了两个声音,一个是256HZ,一个是512HZ

        for(i=0+head.size2/2*(i2-1);i<head.size2/2*i2;i++){

                body[i]=(int)(64*sin(6.28/a*i*i2)+128);//这个波用的是sina函数,以128(ox80)为中心的正弦波,实际上当采样点是0x80的时候是静音,大于则是正的,小于0X80则是负的,前面64则是声音大小,等于128的时候,生成的声音最大


        }

    }
    ofstream ocout;
    ocout.open("123.wav",ios::out|ios::binary);//打开(不存在时生成)123.wav




    ocout.write((char*)&head,sizeof head);//将文件头部分写进文件
    ocout.write((char*)&body,sizeof body);//将数据文件写入程序

    ocout.close();//关闭文件
    cout <<head.samplespersec <<" "<<dou<< endl; // 调试的时候查看一下文件大小是否出错
    return 0;
}
 

调试运行成功