Android音频播放与编码

编程入门 行业动态 更新时间:2024-10-21 18:55:04

Android<a href=https://www.elefans.com/category/jswz/34/1768490.html style=音频播放与编码"/>

Android音频播放与编码

在这篇文章中,我们来学习:

  1. 使用 AudioTrack 进行实时播放
  2. 学习 WAV 格式,将采集得到的数据编码成 WAV 格式
  3. 学习 MediaCodec ,将采集得到的数据编码成 AAC 格式

AudioTrack播放音频数据

Android提供了3套API供我们播放音频:

  1. MediaPlayer:适合在后台长时间播放本地音乐文件或者在线的流式资源,其内部播放音频依赖AudioTrack
  2. SoundPool:适合播放比较短的音频片段,比如游戏声音、按键声、铃声片段等等。
  3. AudioTrack:只能播放PCM数据,更接近底层,使用灵活,支持低延迟播放。

MediaPlayer 和 SoundPool 的使用方法这里不打算讲,下面主要介绍 AudioTrack 的工作流程:

  1. 配置参数, 初始化内部的音频缓冲区
  2. 开始播放
  3. 开启一个工作线程, 不断地向 AudioTrack 的缓冲区“写入”音频数据
  4. 停止播放, 释放资源

配置参数, 初始化内部的音频缓冲区

  1. streamType: 这个参数代表着当前应用使用的哪一种音频管理策略,当系统有多个进程需要播放音频时,这个管理策略会决定最终的展现效果,该参数的可选的值以常量的形式定义在 AudioManager 类中,主要包括:
    1. STREAM_VOCIE_CALL:电话声音
    2. STREAM_SYSTEM:系统声音
    3. STREAM_RING:铃声
    4. STREAM_MUSCI:音乐声
    5. STREAM_ALARM:警告声
    6. STREAM_NOTIFICATION:通知声
  2. sampleRateInHz:音频的采样率
  3. channelConfig:通道数的配置,可选的值以常量的形式定义在 AudioFormat 类中,常用的是 CHANNEL_OUT_MONO(单通道),CHANNEL_OUT_STEREO(双通道)
  4. audioFormat:数据位宽,可选的值也是以常量的形式定义在 AudioFormat 类中,常用的是 ENCODING_PCM_16BIT(16bit),ENCODING_PCM_8BIT(8bit),注意,前者是可以保证兼容所有Android手机的。
  5. bufferSizeInBytesAudioTrack 内部的音频缓冲区的大小,该缓冲区的值不能低于一帧“音频帧”的大小
  6. modeAudioTrack 提供了两种播放模式,一种是 static 方式,一种是 streaming 方式,前者需要一次性将所有的数据都写入播放缓冲区,简单高效,通常用于播放铃声、系统提醒的音频片段; 后者则是按照一定的时间间隔不间断地写入音频数据,理论上它可用于任何音频播放的场景。
  7. private static final int PLAY_MODE = AudioTrack.MODE_STREAM;private AudioTrack createAudioTrack(int streamType, int sampleRate, int channelConfig, int format) {//获得一帧音频帧的缓冲区大小int minBufferSize = AudioTrack.getMinBufferSize(sampleRate, channelConfig, format);if (minBufferSize == AudioTrack.ERROR_BAD_VALUE) {return null;}int audioTrackBufferSize = minBufferSize * 4; //4帧音频帧的大小AudioTrack audioTrack = new AudioTrack(streamType, sampleRate, channelConfig, format, audioTrackBufferSize, PLAY_MODE);if (audioTrack.getState() == AudioTrack.STATE_UNINITIALIZED) {return null;}return audioTrack;
    }

    开始播放

    AudioTrack的播放可以分为两个步骤

  • 1将 PCM 数据写入 AudioTrack 内部
  • 2调用 AudioTrack 的 play 方法进行播放
  • public boolean play(byte[] audioData, int offset, int size) {if (!isStart) {return false;}Log.d(TAG, "play: start");if (audioTrack.write(audioData, offset, size) != size) {Log.d(TAG, "play: size != write size");}audioTrack.play();Log.d(TAG, "play: after play");return true;
    }

    开启一个工作线程, 写入音频数据

  • //AudioCapture是一个音频采集的类,内部会开辟一个工作线程,将采集得到的 PCM 数据通过接口回调出来
    //我们在回调接口取得PCM数据即可
    public class AudioPlayer implements AudioCapture.OnAudioCaptureListener {......public AudioPlayer() {mAudioCapture = new AudioCapture();mAudioCapture.setOnAudioCaptureListener(this);}@Overridepublic void onAudioFrameCaptured(byte[] bytes) {play(bytes, 0, bytes.length);}
    }

    停止播放, 释放资源

  • public void stop() {if (!isStart) {return;}if (audioTrack.getState() == AudioTrack.PLAYSTATE_PLAYING) {audioTrack.stop();}isStart = false;audioTrack.release();audioTrack = null;
    }

    注意: 由于没有做回声和噪声消除,播放的时候最好带上耳机

    将 PCM 数据编码成 WAV 格式

    WAV 是微软开发的一种无损压缩音频文件格式,它的格式比较简单, 只在原始的 PCM 数据上加了一些元信息。 这种文件格式主要分为两个部分:

  • 头文件:主要记录音频文件的一些元信息, 方便播放器等进行识别。例如:采样率、通道数、位宽等
  • 数据块:也就是 PCM 数据
  • 这种文件格式由于没有压缩, 所以音质会比较好,但是文件也会比较大。

    在介绍如何将 PCM 数据编码成 WAV 格式之前,我们先来熟悉这种文件格式。

    WAV 文件头

    WAV 文件主要分为三部分:

  • RIFF
  • fmt
  • data
  • 将 PCM 数据编码成 WAV 格式的过程比较简单,只是多了个写 wav 文件头的步骤。整个过程可以分三个步骤:

  • 先在文件内写入 wav 文件头
  • 将采集得到的 PCM 数据写入文件
  • 利用RandomAccess api 修改 WAV文件头的 ChunkSize 和 Subchunk2Size 字段 (这两个字段在步骤一的写入只是做一个占位的作用,具体数据还是得等到 PCM数据全部写入后才能确定)。
  • 写入wav文件头

    RIFF块

    ChunkID: 取值为“RIFF”
    ChunkSize: 排除ChunkIdChunkSize后,文件剩余的大小
    Format: 固定值为”WAVE“

    fmt 块

    Subchunk1ID: 固定值为“fmt ”; 占4字节(fmt后面还跟着一个空格凑够4字节)
    Subchunk1Size: 固定值为16, 标识fmt块除了Subchunk1ID外占用的字节大小
    AudioFormat: 固定值1
    NumChannels: 音频的通道数
    SampleRate: 音频的采样率
    ByteRate: 1S的音频数据占用的字节数,计算公式:SampleRate * NumChannels * BitsPerSample/8
    BlockAlign: 一个音频采样点占用的字节数, 计算公式:NumChannels * BitsPerSample/8
    BitsPerSample: 音频数据的位宽

    data数据块

    Subchunk2ID:固定为 data
    Subchunk2Size:音频数据的占用的字节大小

    编码

  • 将 PCM 数据编码成 WAV 格式的过程比较简单,只是多了个写 wav 文件头的步骤。整个过程可以分三个步骤:

  • 先在文件内写入 wav 文件头
  • 将采集得到的 PCM 数据写入文件
  • 利用RandomAccess api 修改 WAV文件头的 ChunkSize 和 Subchunk2Size 字段 (这两个字段在步骤一的写入只是做一个占位的作用,具体数据还是得等到 PCM数据全部写入后才能确定)。
  • 写入wav文件头

  • public class WaveWriter {private DataOutputStream dataOutputStream;private int dataLength = 0;private String filePath;public boolean open(String filePath, int sampleRate, short numChannels, short bitsPerSample) throws FileNotFoundException {if (filePath == null) {return false;}this.filePath = filePath;dataOutputStream = new DataOutputStream(new FileOutputStream(filePath));return writeWavHeader(sampleRate, numChannels, bitsPerSample);}private boolean writeWavHeader(int sampleRate, short numChannels, short bitsPerSample{if (dataOutputStream == null) {return false;}WaveHeader waveHeader = new WaveHeader(sampleRate, numChannels, bitsPerSample);try {return waveHeader.writeHeader(dataOutputStream); //先写入文件头} catch (IOException e) {e.printStackTrace();}return false;}......
    }

    写入 PCM 数据

  • public class WaveWriter {private int dataLength = 0; //记录pcm数据的长度public boolean writeData(byte[] pcm, int off, int len) {if (pcm != null && dataOutputStream != null) {try {dataOutputStream.write(pcm, off, len);dataLength += len;return true;} catch (IOException e) {e.printStackTrace();}}return false;}
    }

    修改 ChunkSize 和 Subchunk2Size 字段

  • public class WaveWriter {public boolean closeFile() {boolean ret = false;if (dataOutputStream != null) {try {ret = writeDataSize();} catch (IOException e) {e.printStackTrace();} finally {try {dataOutputStream.close();dataOutputStream = null;} catch (IOException e) {e.printStackTrace();}}}return ret;}private boolean writeDataSize() throws IOException {if (filePath != null) {RandomAccessFile randomAccessFile = new RandomAccessFile(filePath, "rw");randomAccessFile.seek(WaveHeader.RIFF_CHUNK_SIZE_OFFSET);randomAccessFile.write(WaveHelper.int2Bytes(WaveHeader.CHUNKSIZE_EXCLUDE_DATA+ dataLength), 0, 4);randomAccessFile.seek(WaveHeader.DATA_CHUNK_SIZE_OFFSET);randomAccessFile.write(WaveHelper.int2Bytes(dataLength), 0, 4);randomAccessFile.close();return true;}return false;}
    }

    将 PCM 数据编码成 AAC 文件

    MediaCodec 可以帮助我们将 PCM 数据编码成 AAC 格式的数据。

    介绍

    MediaCodec 是 Android 底层多媒体组件之一,我们可以用它来访问底层的编解码器组件。它的工作原理可以用官方的一张图来理解:

    MediaCodec 内部含有两个圆形缓冲区,一个是输入缓冲区,另外一个是输出缓冲区。我们将数据(原始数据或者编码后的数据)填入一个空的输入缓冲区,然后提交给 MediaCodec 。这时,MediaCodec 会进行编解码并将编解码后的数据填入输出缓冲区内。最后我们客户端再请求 MediaCodec 输出缓冲区的数据以获得编解码后的数据。

    简单理解就是一个 input 和 output 的过程,输入数据--> MediaCodec 编解码-->输出数据。

    生命周期

    MediaCodec 的生命周期分为三个:

  • Stopped
  • Executing
  • Released
  • Stopped

    Stopped 内包含三个子状态: UninitializedConfigured 和 Error。当我们用MediaCodec 的工厂方法创建一个实例时,MediaCodec 处于 Uninitialized 状态。得到实例后,调用 MediaCodec.configure(...)MediaCodec 进入 Configured 状态。当MediaCodec 处理数据发生错误时会转移到 Error 状态。

    Executing

    Executing内部也分为三个状态: FlushedRunning 和 End-of-Stream。一旦调用MediaCodec.start 后,MediaCodec便从Configure状态进入Flushed状态。处于这个状态的MediaCodec,其内部的输入和输出缓冲区都是空闲的。一旦外部向MediaCodec请求输入缓冲区填数据时,MediaCodec进入Running状态。而当外部在输入缓冲区填入end-of-stream标记时,MediaCodec内部状态将转移到End-of-Stream状态,在这个状态的MediaCodec,不再接收新的请求。

    当处于MediaCodec处于Executing状态时,调用它的flush可以令MediaCodec返回Flushed子状态。

    Released

    MediaCodec完成它的工作后,应该调用MediaCodec.release()释放掉它内部的资源。

    使用

    上述只是对MediaCodec的一个简短介绍,详细内容可以参考官方文档。下面我们来看看MediaCodec的一个使用套路。总体来说,MediaCodec的使用步骤可以分为三个:

  • 创建实例
  • 配置MediaCodec, 并调用它的start方法
  • 数据处理

创建实例

MediaCodec提供了两个静态方法给我们创建实例,一个是createEncoderByType(mineType),另外一个是createByCodecName(name)。前者是使用一个mineType创建MediaCodec,由于设备可能不支持mimeType对应的编解码器,所以不推荐使用这种方式来创建。建议使用createByCodecName(name)MediaCodecList.findEncoderForFormat创建实例。这种方式与前面的一种不同,它会首先判断当前设备是否支持指定的编解码器,如果支持的话,我们可以创建MediaCodec。如果不支持的话,我们再提示给用户。

  • public static final int DEFAULT_BIT_RATE = 128 * 1024; //128kbpublic static final int DEFAULT_SIMPLE_RATE = 44100; //44100Hzpublic static final int DEFAULT_CHANNEL_COUNTS = 1; //单通道public static final int DEFAULT_MAX_INPUT_SIZE = 16384; //16kprivate MediaCodec createMediaCodec() {mediaFormat = createMediaFormat();MediaCodecList mediaCodecList = new MediaCodecList(MediaCodecList.ALL_CODECS);String name = mediaCodecList.findEncoderForFormat(mediaFormat);if (name != null) {try {return MediaCodec.createByCodecName(name);} catch (IOException e) {e.printStackTrace();}}return null; //不支持对应的编解码器
    }private MediaFormat createMediaFormat() {//mimeType设置为AACMediaFormat mediaFormat=MediaFormat.createAudioFormat(MediaFormat.MIMETYPE_AUDIO_AAC, DEFAULT_SIMPLE_RATE, DEFAULT_CHANNEL_COUNTS);mediaFormat.setInteger(MediaFormat.KEY_AAC_PROFILE,MediaCodecInfo.CodecProfileLevel.AACObjectLC);mediaFormat.setInteger(MediaFormat.KEY_BIT_RATE, DEFAULT_BIT_RATE);mediaFormat.setInteger(MediaFormat.KEY_MAX_INPUT_SIZE, DEFAULT_MAX_INPUT_SIZE);return mediaFormat;
    }

    配置MediaCodec, 并调用它的start方法

    这个步骤比较简单,只需要调用对应的方法即可

  • public void start() {configure();mediaCodec.start();
    }private void configure() {mediaCodec = createMediaCodec();if (mediaCodec == null) {throw new IllegalStateException("该设备不支持AAC编码器");}//configure函数的第二个参数如果传入Surface, 表示将数据通过Surface渲染出来, 传入null表示可以//利用ByteBuffer来获得数据mediaCodec.configure(mediaFormat, null, null, MediaCodec.CONFIGURE_FLAG_ENCODE);
    }

    数据处理

    MediaCodec的数据处理分为两种模式: 同步模式和异步模式。其中异步模式是API为21之后引入的。不管是同步还是异步模式,MediaCodec数据处理的思想都是一样的:

  • MediaCode请求空闲的输入缓冲区。在同步模式中,调用dequeueInput获得可用的输入缓冲区;在异步模式中,可以在MediaCodec.Callback.onInputBufferAvailable(…)回调接口获取空闲的输入缓冲区。
  • 向缓冲区填入数据,接着调用queueInputBuffer(...)将缓冲区提交给MediaCode
  • MediaCode请求编解码后的输出缓冲区。在同步模式中,调用dequeueOutputBuffer获得可用的输出缓冲区;在异步模式中,我们可以在MediaCodec.Callback.onOutputBufferAvailable(…) 回调接口中获得可用的输出缓冲区。
  • 从输出缓冲区取出数据,最后调用releaseOutputBuffer方法释放缓冲区。

同步模式

/*** 将采集得到的数据提交给MediaCodec*/
public synchronized void encode(MediaCodec mediaCodec, byte[] data) {if (!isStart) {return;}if (data != null) {int bufferIndexId = mediaCodec.dequeueInputBuffer(10000);if (bufferIndexId >= 0) {ByteBuffer inputBuffer = mediaCodec.getInputBuffer(bufferIndexId);inputBuffer.clear();inputBuffer.put(data);mediaCodec.queueInputBuffer(bufferIndexId, 0, data.length, System.nanoTime() / 1000,0);}} 
}/*** 从MediaCodec中取得已经编码好的数据, 并写入文件*/
private synchronized void queryEncodedData(MediaCodec mediaCodec) {if (mediaCodec == null) {return;}MediaCodec.BufferInfo bufferInfo = new MediaCodec.BufferInfo();int outputBufferIndexId = mediaCodec.dequeueOutputBuffer(bufferInfo, 10000);if (outputBufferIndexId >= 0) {ByteBuffer byteBuffer = mediaCodec.getOutputBuffer(outputBufferIndexId);byteBuffer.position(bufferInfo.offset);byteBuffer.limit(bufferInfo.offset + bufferInfo.size);byte[] frame = new byte[bufferInfo.size];byteBuffer.get(frame, 0, bufferInfo.size);writeToFile(frame);mediaCodec.releaseOutputBuffer(outputBufferIndexId, false);}
}private void writeToFile(byte[] frame) {byte[] packetWithADTS = new byte[frame.length + 7];System.arraycopy(frame, 0, packetWithADTS, 7, frame.length);//必须在裸aac流上添加ADTS头, 不然播放器播放不了addADTStoPacket(packetWithADTS, packetWithADTS.length);if (dataOutputStream != null) {try {dataOutputStream.write(packetWithADTS, 0, packetWithADTS.length);} catch (IOException e) {e.printStackTrace();}}
}private void addADTStoPacket(byte[] packet, int packetLen) {int profile = 2;  //AAC LC,MediaCodecInfo.CodecProfileLevel.AACObjectLC;int freqIdx = 4;  //44100int chanCfg = DEFAULT_CHANNEL_COUNTS; //默认单声道// fill in ADTS datapacket[0] = (byte) 0xFF;packet[1] = (byte) 0xF9;packet[2] = (byte) (((profile - 1) << 6) + (freqIdx << 2) + (chanCfg >> 2));packet[3] = (byte) (((chanCfg & 3) << 6) + (packetLen >> 11));packet[4] = (byte) ((packetLen & 0x7FF) >> 3);packet[5] = (byte) (((packetLen & 7) << 5) + 0x1F);packet[6] = (byte) 0xFC;
}
  • 异步模式

    异步模式是5.0后引入的,在调用configure之前,给MediaCode设置mediaCodec.setCallback(...)接口。这样,当MediaCodec内部缓冲区可用时,会自动回调相应的接口,我们只需要在接口中处理数据即可。

  • private class AsyEncodeCallback extends MediaCodec.Callback {//输入缓冲区可用时, 回调这个接口@Overridepublic void onInputBufferAvailable(@NonNull MediaCodec codec, int index) {if (!isStart) {try {innerStop(); //停止编码} catch (IOException e) {e.printStackTrace();}return;}byte[] buf = new byte[2 * 1024];int ret = mAudioCapture.readData(buf, 0, buf.length); //从AudioRecord读取数据if (ret > 0) {ByteBuffer byteBuffer = codec.getInputBuffer(index);byteBuffer.clear();byteBuffer.put(buf);mediaCodec.queueInputBuffer(index, 0, buf.length, System.nanoTime() / 1000, 0);}}//输出缓冲区可用时, 回调这个接口@Overridepublic void onOutputBufferAvailable(@NonNull MediaCodec codec, int index, @NonNull MediaCodec.BufferInfo info) {if (!isStart) {try {innerStop();} catch (IOException e) {e.printStackTrace();}return;}ByteBuffer byteBuffer = mediaCodec.getOutputBuffer(index);byteBuffer.position(info.offset);byteBuffer.limit(info.offset + info.size);byte[] frame = new byte[info.size];byteBuffer.get(frame, 0, info.size);writeToFile(frame);mediaCodec.releaseOutputBuffer(index, false);}@Overridepublic void onError(@NonNull MediaCodec codec, @NonNull MediaCodec.CodecException e) {}@Overridepublic void onOutputFormatChanged(@NonNull MediaCodec codec, @NonNull MediaFormat format) {}
    }

    总结

    这次我们主要学习了如何利用 AudioTrack来播放 PCM数据以及将 PCM 数据编码成 WAV和 AAC 格式进行保存。

 

更多推荐

Android音频播放与编码

本文发布于:2023-07-28 16:06:38,感谢您对本站的认可!
本文链接:https://www.elefans.com/category/jswz/34/1244982.html
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。
本文标签:音频   Android

发布评论

评论列表 (有 0 条评论)
草根站长

>www.elefans.com

编程频道|电子爱好者 - 技术资讯及电子产品介绍!