作为学习C#的借口,我一直在尝试编写一个简单的项目:创建音频文件。首先,我想确保我可以编写符合WAVE格式的文件。我已经在线研究了格式(例如,here),但每当我尝试播放文件时,它都无法正常打开。这是我的代码。缺少或不正确的东西?
uint numsamples = 44100;
ushort numchannels = 1;
ushort samplelength = 1; // in bytes
uint samplerate = 22050;
FileStream f = new FileStream("a.wav", FileMode.Create);
BinaryWriter wr = new BinaryWriter(f);
wr.Write("RIFF");
wr.Write(36 + numsamples * numchannels * samplelength);
wr.Write("WAVEfmt ");
wr.Write(16);
wr.Write((ushort)1);
wr.Write(numchannels);
wr.Write(samplerate);
wr.Write(samplerate * samplelength * numchannels);
wr.Write(samplelength * numchannels);
wr.Write((ushort)(8 * samplelength));
wr.Write("data");
wr.Write(numsamples * samplelength);
// for now, just a square wave
Waveform a = new Waveform(440, 50);
double t = 0.0;
for (int i = 0; i < numsamples; i++, t += 1.0 / samplerate)
{
wr.Write((byte)((a.sample(t) + (samplelength == 1 ? 128 : 0)) & 0xff));
}
答案 0 :(得分:8)
主要问题是:
BinaryWriter.Write(string)
会为BinaryReader
写一个前缀为的字符串,以便将其读回。它不打算像你的情况一样使用。您需要直接编写字节而不是使用BinaryWriter.Write(string)
。
你应该做什么:
将字符串转换为字节,然后直接写入字节。
byte[] data = System.Text.Encoding.ASCII.GetBytes("RIFF");
binaryWriter.Write(data);
或使其成为一行:
binaryWriter.Write(System.Text.Encoding.ASCII.GetBytes("RIFF"));
可能还有其他问题,例如您编写的整数可能与所需的大小不同。你应该仔细检查一下。
至于endianess,你输入的链接表明数据是little-endian而BinaryWriter
使用little-endian,所以这应该不是问题。
答案 1 :(得分:2)
最简单的方法,你可以简单地改变:
wr.Write("RIFF");
为:
wr.Write("RIFF".ToArray());
在二进制文件中写入一个字符串,它将包含字符串的长度,以便以后可以将其反序列化为字符串。在这种情况下,您只需要将四个字节写为四个字节,并将其转换为char数组即可。
答案 2 :(得分:0)
我缺少正确的WAV数据,但尝试使用此代码替换代码中生成标题的部分(适当替换):
wr.Write(Encoding.ASCII.GetBytes("RIFF"));
wr.Write(0);
wr.Write(Encoding.ASCII.GetBytes("WAVE"));
wr.Write(Encoding.ASCII.GetBytes("fmt "));
wr.Write(18 + (int)(numsamples * samplelength));
wr.Write((short)1); // Encoding
wr.Write((short)numchannels); // Channels
wr.Write((int)(samplerate)); // Sample rate
wr.Write((int)(samplerate * samplelength * numchannels)); // Average bytes per second
wr.Write((short)(samplelength * numchannels)); // block align
wr.Write((short)(8 * samplelength)); // bits per sample
wr.Write((short)(numsamples * samplelength)); // Extra size
wr.Write("data");
答案 3 :(得分:0)
@ Alvin-wong的回答非常完美。只是想添加另一个建议,虽然还有几行:
binaryWriter.Write('R');
binaryWriter.Write('I');
binaryWriter.Write('F');
binaryWriter.Write('F');