在安卓系统中混合音频

5
我尝试了解这个链接:http://mobilengineering.blogspot.com/2012/06/audio-mix-and-record-in-android.html?showComment=1369622288028#c2333829870074273419。但是在混合音频文件后,位于SD卡上的文件mixed.wav无法播放,我不知道为什么。您能帮帮我吗?非常感谢。
以下是我的代码: public class MainActivity extends Activity {
 public static final int FREQUENCY = 44100;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    try {
        mixSound();
    } catch (IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

private void mixSound() throws IOException {
    AudioTrack audioTrack = new AudioTrack(AudioManager.STREAM_MUSIC, 44100, AudioFormat.CHANNEL_OUT_STEREO, AudioFormat.ENCODING_PCM_16BIT, 44100, AudioTrack.MODE_STREAM);

    InputStream in1 = getResources().openRawResource(R.raw.media_b);
    InputStream in2 = getResources().openRawResource(R.raw.media_c);

    byte[] arrayMusic1 = null;
    arrayMusic1 = new byte[in1.available()];
    arrayMusic1 = createMusicArray(in1);
    in1.close();

    byte[] arrayMusic2 = null;
    arrayMusic2 = new byte[in2.available()];
    arrayMusic2 = createMusicArray(in2);
    in2.close();


    byte[] output = new byte[arrayMusic1.length];

    audioTrack.play();

    for (int i = 0; i < output.length; i++) {
        float samplef1 = arrayMusic1[i] / 128.0f; 
        float samplef2 = arrayMusic2[i] / 128.0f;
        float mixed    = samplef1 + samplef2;

        // reduce the volume a bit:
        mixed *= 0.8;
        // hard clipping
        if (mixed > 1.0f)  mixed = 1.0f;
        if (mixed < -1.0f) mixed = -1.0f;

        byte outputSample = (byte) (mixed * 128.0f);
        output[i]         = outputSample;   
    }

    audioTrack.write(output, 0, output.length);
    convertByteToFile(output);
}

public static byte[] createMusicArray(InputStream is) throws IOException {

    ByteArrayOutputStream baos = new ByteArrayOutputStream();
    byte[] buff = new byte[10240];
    int i = Integer.MAX_VALUE;
    while ((i = is.read(buff, 0, buff.length)) > 0) {
        baos.write(buff, 0, i);
    }

    return baos.toByteArray(); // be sure to close InputStream in calling function

}

public static void convertByteToFile(byte[] fileBytes) throws FileNotFoundException {

    BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(Environment.getExternalStorageDirectory().getPath()+"/mixed.wav"));
    try {
        bos.write(fileBytes);
        bos.flush();
        bos.close();    
    } catch (IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

}

1个回答

3
你正在输出的只是PCM数据。一个有效的WAV文件还需要一个头部:
Offset    Size  Name           Description
------------------------------------------------------------------------
0         4     ChunkID        Contains the letters "RIFF" in ASCII form
                               (0x52494646 big-endian form).
4         4     ChunkSize      36 + SubChunk2Size, or more precisely:
                               4 + (8 + SubChunk1Size) + (8 + SubChunk2Size)
                               This is the size of the rest of the chunk 
                               following this number.  This is the size of the 
                               entire file in bytes minus 8 bytes for the
                               two fields not included in this count:
                               ChunkID and ChunkSize.
8         4     Format         Contains the letters "WAVE"
                               (0x57415645 big-endian form).

12        4     Subchunk1ID    Contains the letters "fmt "
                               (0x666d7420 big-endian form).
16        4     Subchunk1Size  16 for PCM.  This is the size of the
                               rest of the Subchunk which follows this number.
20        2     AudioFormat    PCM = 1 (i.e. Linear quantization)
                               Values other than 1 indicate some 
                               form of compression.
22        2     NumChannels    Mono = 1, Stereo = 2, etc.
24        4     SampleRate     8000, 44100, etc.
28        4     ByteRate       == SampleRate * NumChannels * BitsPerSample/8
32        2     BlockAlign     == NumChannels * BitsPerSample/8
                               The number of bytes for one sample including
                               all channels. I wonder what happens when
                               this number isn't an integer?
34        2     BitsPerSample  8 bits = 8, 16 bits = 16, etc.
          2     ExtraParamSize if PCM, then doesn't exist
          X     ExtraParams    space for extra parameters

36        4     Subchunk2ID    Contains the letters "data"
                               (0x64617461 big-endian form).
40        4     Subchunk2Size  == NumSamples * NumChannels * BitsPerSample/8
                               This is the number of bytes in the data.
                               You can also think of this as the size
                               of the read of the subchunk following this 
                               number.

在此之后,您编写PCM数据。

(参考资料)。


如何修复它,我不知道怎么做。 你能帮我吗?非常感谢 :) - Noe Love
你需要编写头文件中指定的所有字段。首先是字符串“RIFF”,然后是一个包含“36+你的音频数据大小”的32位值,接着是字符串“WAVE”,以此类推。 - Michael
非常感谢。我有些成功,但mixed.wav文件中有噪音。如何修复它并去除白噪声?我的wav字节数组该怎么办?你能帮我吗? - Noe Love
你应该在头部之后立即编写PCM数据。您可以通过将其转储到没有任何头文件的文件中,然后在PC上作为原始PCM数据导入Audacity来双重检查您的PCM数据是否正确。 - Michael
我改成了short[]后,它的表现非常糟糕:( 我不知道哪里出了问题。 - Noe Love
显示剩余3条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接