解码音频文件并重新编码为所需的PCM格式:44,100 kHz,2通道,16位

poc*_*cmo 7 audio android pcm mediacodec mediaextractor

我想将音频文件解码为原始PCM数据,以将其流式传输到本地网络中的播放设备.我使用了引入API级别16 的新MediaExtractorMediaCodec类.该器件要求PCM数据为44,100 kHz,有2个通道和16位样本大小.只要输入文件大致符合这些要求,这就可以正常工作.但是,无论何时我正在解码使用的文件 - 例如 - 32,000 kHz的采样率并且可能只有一个通道,那么我无法从MediaCodec类获得所需的输出.

因为似乎无法指定MediaCodec类的输出格式.所以我决定实例化另一个MediaCodec对象,将原始数据重新编码为我想要的格式.根据Android 4.1 支持的媒体格式列表,Android支持编码到PCM/Wave.但是我无法创建编码为PCM/Wave的MediaCodec对象.我尝试将各种MIME类型传递给MediaCodec.createEncoderByType(type); 但我总是以IOException失败:

java.io.IOException: Failed to allocate component instance
at android.media.MediaCodec.native_setup(Native Method)
at android.media.MediaCodec.<init>(MediaCodec.java:210)
at android.media.MediaCodec.createEncoderByType(MediaCodec.java:194)
[..]
Run Code Online (Sandbox Code Playgroud)

有没有人能够成功创建编码为PCM/Wave的MediaCodec实例并且可以为我提供一个工作示例?

Dav*_*ave 6

根据我们在评论中的对话,这个答案涉及使用OpenSL将音频数据解码为PCM.不幸的是,我无法使用该MediaCodec课程提供类似的答案.

首先,设置一个Android NDK项目(使用Eclipse:右键单击项目,Android工具 - >添加本机支持...).然后,在创建的Android.mk文件中,至少需要链接OpenSL库:

LOCAL_LDLIBS += -lOpenSLES
Run Code Online (Sandbox Code Playgroud)

如果您对NDK项目不熟悉,那么旧的互联网上会有很多教程,例如,这里这里.

一旦你有一个NDK项目工作,总的目标是设置一个音频播放器作为PCM的解码器.实际上NDK样本中有一个例子就是这样.它比你制作一个最小功能的工具要复杂一点,但它应该让你前进.关于此链接发生的事情有一个非常容易理解的描述(搜索"将音频解码为PCM").我在评论中为您提供了相关OpenSL规范的链接,但它又是.SLDataFormat_PCM指定数据接收器时需要使用的结构如下所示:

SLDataFormat_PCM pcm = {
    SL_DATAFORMAT_PCM,
    2,                            // numChannels
    SL_SAMPLINGRATE_44_1,         // samplesPerSec
    SL_PCMSAMPLEFORMAT_FIXED_16,  // bitsPerSample
    SL_PCMSAMPLEFORMAT_FIXED_16,  // containerSize
    SL_SPEAKER_FRONT_LEFT |
        SL_SPEAKER_FRONT_RIGHT,   // channelMask
    SL_BYTEORDER_LITTLEENDIAN     // endianness
};
Run Code Online (Sandbox Code Playgroud)

按照示例和描述性指南,您将使用Android简单缓冲区队列数据定位器和上述数据格式定义SLDataSink.然后,播放器应为您提供一系列包含正确格式数据的缓冲区.然后,您可以将该数据传递回Java,或者(更好地)从本机代码传输数据.