audiotrack:播放原始pcm 16bit wav文件的噪音

时间:2018-04-15 05:34:02

标签: android audio pcm

我正在尝试播放大小为230mb和20分钟的wav文件,其属性如下:

ffmpeg -i 1.wav

Stream #0:0: Audio: pcm_s16le ([1][0][0][0] / 0x0001), 44100 Hz, stereo, s16, 1411 kb/s

我正在学习如何使用audiotrack。

我找到了两个使用audiotrack播放音频播放的解决方案。

解决方案1:以下播放音频

int frequency = 44100;
int channelConfiguration = AudioFormat.CHANNEL_CONFIGURATION_STEREO;
int audioEncoding = AudioFormat.ENCODING_PCM_16BIT;

int bufferSize = AudioTrack.getMinBufferSize(frequency, channelConfiguration,audioEncoding);

AudioTrack audioTrack = new AudioTrack(AudioManager.STREAM_MUSIC, frequency,
        channelConfiguration, audioEncoding, bufferSize,
        AudioTrack.MODE_STREAM);

int count = 0;
byte[] data = new byte[bufferSize];
try{
    FileInputStream fileInputStream = new FileInputStream(listMusicFiles.get(0).listmusicfiles_fullfilepath);
    DataInputStream dataInputStream = new DataInputStream(fileInputStream);
    audioTrack.play();

    while((count = dataInputStream.read(data, 0, bufferSize)) > -1){
        audioTrack.write(data, 0, count);
    }

    audioTrack.stop();
    audioTrack.release();
    dataInputStream.close();
    fileInputStream.close();
}
catch (FileNotFoundException e){
    e.printStackTrace();
} catch (IOException e) {
    e.printStackTrace();
}

第二种解决方案:播放噪音

int frequency = 44100;
int channelConfiguration = AudioFormat.CHANNEL_CONFIGURATION_STEREO;
int audioEncoding = AudioFormat.ENCODING_PCM_16BIT;

int bufferSize = AudioTrack.getMinBufferSize(frequency, channelConfiguration,audioEncoding);

short[] audiodata = new short[bufferSize];

try {

    DataInputStream dis = new DataInputStream(
            new BufferedInputStream(new FileInputStream(
                    listMusicFiles.get(0).listmusicfiles_fullfilepath)));

    AudioTrack audioTrack = new AudioTrack(AudioManager.STREAM_MUSIC, frequency,
            channelConfiguration, audioEncoding, bufferSize,
            AudioTrack.MODE_STREAM);

    audioTrack.play();

    while (dis.available() > 0) {
        int i = 0;
        while (dis.available() > 0 && i < audiodata.length) {
            audiodata[i] = dis.readShort();
            i++;
        }
        audioTrack.write(audiodata, 0, audiodata.length);
    }

    dis.close();
} catch (Throwable t) {
    Log.e("AudioTrack", "Playback Failed");
}

我是新的短样本和字节样本。我试图理解,但不是那么容易。

我能理解第一个解决方案是使用字节样本,第二个解决方案是使用短样本。

那么为什么第二种解决方案无效。

1 个答案:

答案 0 :(得分:0)

short类型的默认大小为two bytes。您也可以查看this documentation

音频轨道具有推荐的缓冲区大小和采样率,可以使用this answer中建议的方式找到。请看看。

但是,使用建议的采样率(在您的情况下为44100 Hz)和使用以下代码段获得的建议缓冲区大小来播放音频非常重要。

AudioTrack.getMinBufferSize(frequency, channelConfiguration, audioEncoding) 

在使用short数组的实现中,缓冲区大小加倍,因此在播放音频时会产生噪音。我建议,您可以考虑使用short在实现中将大小除以2来更改缓冲区大小。

short[] audiodata = new short[(int) bufferSize / 2];

希望你已经理解了这个问题。