C 输出 Wav 文件没有产生任何声音

Posted

技术标签:

【中文标题】C 输出 Wav 文件没有产生任何声音【英文标题】:C output Wav file didn't produce any sound 【发布时间】:2014-12-26 12:29:29 【问题描述】:

我尝试创建一个能产生 10 秒 C 音符的 C 代码。但似乎输出的 .wav 文件没有产生任何声音。

我还是 C 编程的新手,如果你能指出我的错误会很有帮助。

#include <stdio.h>
#include <math.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>

//music note
#define C 261.6256

#define TIME 10
#define POINT 20
#define AMP 10000
#define c 5

//wav file header
typedef struct

    char ChuckID[4];
    unsigned long ChuckSize;
    char format[4];
    char subChunk1ID[4];
    unsigned long SubChunk1Size;
    unsigned short AudioFormat;
    unsigned short NumChannels;
    unsigned long SampleRate;
    unsigned long ByteRate;
    unsigned short block_allign;
    unsigned short bits_per_sample;
    char data[4];
    unsigned long data_size;

    /*char  riff_tag[4];
    int     riff_length;
    char    wave_tag[4];
    char    fmt_tag[4];
    int     fmt_length;
    short   audio_format;
    short   num_channels;
    int     sample_rate;
    int     byte_rate;
    short   block_align;
    short   bits_per_sample;
    char    data_tag[4];
    int     data_length;*/
 wavheader;

int main(int argc, char **argv)

    wavheader wave = "RIFF",1764036,"WAVE","fmt",16,1,1,44100,176400,4,32,"data",1764000;

    float data;
    float f = C;
    int fs = 44100;
    int k;
    float *buff;

    FILE *out_file = fopen("ongaku.wav","w");
    buff = (float*)malloc(sizeof(float)*fs*TIME);

    for (k = 0; k<(int)(TIME*fs); k++)
    
        data=AMP*sin(2*M_PI*f*k/fs);
        //printf("%f\n",data);
    

    fwrite(buff,sizeof(float),fs*TIME,out_file);

    return 0;

【问题讨论】:

您尚未将文件头写入文件或将任何数据写入buff 【参考方案1】:

我可以使用 8 位数据进行此操作,但使用 12/16 位数据不成功,更不用说 float 数据了。重要的一件事是不要在标头中硬编码缓冲区大小。其他需要注意的点是字节序(我碰巧不需要调整)和结构包装(同上)。在处理 12 位数据时,我对 BPS/8 的使用也会变得不稳定。

#include <stdio.h>
#include <math.h>
#include <stdlib.h>

#define FREQ    261.6256    // C
//#define FREQ    440.0       // A
#define M_PI    3.14159265358979323846
#define TIME    10
#define AMP     64.0        // don't use max volume
#define MID     128.0       // 8-bit is range 0..255
//#define MID   0.0         // 16-bit is range -32767.. 32767
#define BPS     8
#define CHANNS  1
#define RATE    44100

//wav file header
typedef struct 
    char ChuckID[4];
    unsigned long ChuckSize;
    char format[4];
    char subChunk1ID[4];
    unsigned long SubChunk1Size;
    unsigned short AudioFormat;
    unsigned short NumChannels;
    unsigned long SampleRate;
    unsigned long ByteRate;
    unsigned short block_allign;
    unsigned short bits_per_sample;
    char data[4];
    unsigned long data_size;
 wavheader;

int main(int argc, char **argv)

    int k, samples = RATE * TIME;
    double data;
    FILE *out_file;
    unsigned char *buff;
    wavheader wave = 
        "RIFF",
        36 + samples * CHANNS * BPS/8,
        "WAVE",
        "fmt ",         // "fmt" was error in OP
        16, 
        1,
        CHANNS,
        RATE,
        RATE * CHANNS * BPS/8,
        CHANNS * BPS/8,
        BPS,
        "data",
        samples * CHANNS * BPS/8 
        ;

    buff = malloc(BPS/8 * samples);
    out_file = fopen("ongaku.wav","w");
    fwrite(&wave, sizeof(wave), 1, out_file);
    for (k=0; k<samples; k++) 
        data = MID + AMP * sin(2 * M_PI * FREQ * TIME * k / (double)samples);
        buff[k] = (unsigned char)floor(data+0.5);
    
    fwrite(buff, BPS/8, samples, out_file);
    fclose (out_file);
    free (buff);
    return 0;

【讨论】:

@Kamilin 谢谢。我发现使用最大振幅会产生令人不快的声音。抱歉,我不知道为什么我无法让 16 位工作。【参考方案2】:

将一些数据放入 buff 中,我猜你的数据变量持有那个值。在那之后 如果其他一切正常,请使用

fflush(out_file);

或使用

fclose(out_file);

【讨论】:

以上是关于C 输出 Wav 文件没有产生任何声音的主要内容,如果未能解决你的问题,请参考以下文章

如何将扬声器输出写入文件 sounddevice

如何将 MP3 输出到声卡

MediaPlayer 和 AudioTrack 不输出相同的声音

为啥将拆分为 wav 文件的旋律转换为拆分的 mp3 会在片段边界处产生不好的声音?

使用 C 绕过声音/音频文件 (WAV)

怎么将8位的WAV声音文件转换为16位?