我正在尝试将 2 个 WAV 文件混合成一个 WAV 文件。这些文件将始终具有完全相同的持续时间并具有相同的格式(16 位、44.1 kHz、有符号、小端、单声道)。使用 AudioSystem.getAudioInputSream 的 ByteArrayOutputStream 将两个 WAV 放入字节数组,以确保我只获取 PCM 数据而没有 header 。
在其他几个线程的帮助下,我已经能够成功地组合阵列,但并非没有向信号引入大量噪声。这听起来绝对不像削波或失真,但为了安全起见,我尝试对每对字节的总和进行平均,这只会让一切变得更安静,噪音和所有。
任何见解将不胜感激!我已经尝试了两种应该做同样事情的方法,并且似乎产生了相同的声音结果。
第一种,更简单的方法:
private byte[] mixBuffers(byte[] bufferA, byte[] bufferB) {
byte[] array = new byte[bufferA.length];
for (int i=0; i<bufferA.length; i++) {
array[i] = (byte) ((bufferA[i] + bufferB[i]));
}
return array;
}
第二种更具体的方法:
private byte[] mixBuffers(byte[] bufferA, byte[] bufferB) {
byte[] array = new byte[bufferA.length];
for (int i=0; i<bufferA.length; i+=2) {
short buf1A = bufferA[i+1];
short buf2A = bufferA[i];
buf1A = (short) ((buf1A & 0xff) << 8);
buf2A = (short) (buf2A & 0xff);
short buf1B = bufferB[i+1];
short buf2B = bufferB[i];
buf1B = (short) ((buf1B & 0xff) << 8);
buf2B = (short) (buf2B & 0xff);
short buf1C = (short) (buf1A + buf1B);
short buf2C = (short) (buf2A + buf2B);
short res = (short) (buf1C | buf2C);
array[i] = (byte) res;
array[i+1] = (byte) (res >> 8);
}
return array;
}
如果有帮助,下面是我如何将文件放入字节数组:
{
ByteArrayOutputStream baos = new ByteArrayOutputStream();
AudioInputStream ais;
ais = AudioSystem.getAudioInputStream(audioFile);
int read;
byte[] buffer = new byte[1024];
while ((read = ais.read(buffer)) != -1) {
baos.write(buffer, 0, read);
}
baos.flush();
byte[] byteBufferA = baos.toByteArray();
baos = new ByteArrayOutputStream();
ais = AudioSystem.getAudioInputStream(audioFile2);
buffer = new byte[1024];
while ((read = ais.read(buffer)) != -1) {
baos.write(buffer, 0, read);
}
baos.flush();
byte[] byteBufferB = baos.toByteArray();
byte[] byteBufferC = mixBuffers(byteBufferA, byteBufferB);
}
最佳答案
这些文件包含 16 位样本 - 每个样本都是两个字节长。但是,您的第一种方法尝试独立混合每个字节。结果就像通过添加每个数字(不带进位)来添加一个数字:165 + 248 -> 103。
您的第二种方法是使用按位或而不是加法。这就像通过取每个数字的最大值将两个数字相加:165 + 248 -> 268。
尝试使用第二种方法,但将 |
替换为 +
。
关于Java:混合两个 WAV 文件而不引入噪音,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32856836/