一、什么是AudioTrack

/*** The AudioTrack class manages and plays a single audio resource for Java applications.* It allows streaming of PCM audio buffers to the audio sink for playback. This is* achieved by "pushing" the data to the AudioTrack object using one of the*  {@link #write(byte[], int, int)}, {@link #write(short[], int, int)},*  and {@link #write(float[], int, int, int)} methods.** <p>An AudioTrack instance can operate under two modes: static or streaming.<br>* In Streaming mode, the application writes a continuous stream of data to the AudioTrack, using* one of the {@code write()} methods. These are blocking and return when the data has been* transferred from the Java layer to the native layer and queued for playback. The streaming* mode is most useful when playing blocks of audio data that for instance are:** <ul>*   <li>too big to fit in memory because of the duration of the sound to play,</li>*   <li>too big to fit in memory because of the characteristics of the audio data*         (high sampling rate, bits per sample ...)</li>*   <li>received or generated while previously queued audio is playing.</li>* </ul>** The static mode should be chosen when dealing with short sounds that fit in memory and* that need to be played with the smallest latency possible. The static mode will* therefore be preferred for UI and game sounds that are played often, and with the* smallest overhead possible.** <p>Upon creation, an AudioTrack object initializes its associated audio buffer.* The size of this buffer, specified during the construction, determines how long an AudioTrack* can play before running out of data.<br>* For an AudioTrack using the static mode, this size is the maximum size of the sound that can* be played from it.<br>* For the streaming mode, data will be written to the audio sink in chunks of* sizes less than or equal to the total buffer size.** AudioTrack is not final and thus permits subclasses, but such use is not recommended.*/

二、AudioTrack 构造方法参数介绍

 public AudioTrack(int streamType, int sampleRateInHz, int channelConfig, int audioFormat,int bufferSizeInBytes, int mode)throws IllegalArgumentException {this(streamType, sampleRateInHz, channelConfig, audioFormat,bufferSizeInBytes, mode, AudioManager.AUDIO_SESSION_ID_GENERATE);}
streamType:  Android将系统的声音分为好几种流类型,下面是几个常见的: ·  STREAM_ALARM:警告声·  STREAM_MUSIC:音乐声,例如music等·  STREAM_RING:铃声·  STREAM_SYSTEM:系统声音,例如低电提示音,锁屏音等·  STREAM_VOCIE_CALL:通话声sampleRateInHz: 采样率 (MediaRecoder 的采样率通常是8000Hz AAC的通常是44100Hz。* 设置采样率为44100,目前为常用的采样率,官方文档表示这个值可以兼容所有的设置)channelConfig:指定捕获音频的声道数目。在AudioFormat类中指定用于此的常量 audioFormat :指定音频量化位数 ,在AudioFormaat类中指定了以下各种可能的常量。* 通常我们选择ENCODING_PCM_16BIT和ENCODING_PCM_8BIT PCM代表的是脉冲编码调制,它实际上是原始音频样本。* 因此可以设置每个样本的分辨率为16位或者8位,16位将占用更多的空间和处理能力,表示的音频也更加接近真实。bufferSizeInBytes:指定缓冲区大小,调用AudioTrack类的getMinBufferSize方法可以获得,mode : AudioTrack有两种数据加载模式(MODE_STREAM和MODE_STATIC)MODE_STREAM:在这种模式下,通过write一次次把音频数据写到AudioTrack中。这和平时通过write系统调用往文件中写数据类似,但这种工作方式每次都需要把数据从用户提供的Buffer中拷贝到AudioTrack内部的Buffer中,这在一定程度上会使引入延时。为解决这一问题,AudioTrack就引入了第二种模式。MODE_STATIC:这种模式下,在play之前只需要把所有数据通过一次write调用传递到AudioTrack中的内部缓冲区,后续就不必再传递数据了。这种模式适用于像铃声这种内存占用量较小,延时要求较高的文件。但它也有一个缺点,就是一次write的数据不能太多,否则系统无法分配足够的内存来存储全部数据。

三、AudioTrack 两种模式使用

3.1、AudioTrack MODE_STATIC 使用

this.audioTrack = new AudioTrack(AudioManager.STREAM_MUSIC, 44100,AudioFormat.CHANNEL_OUT_STEREO, AudioFormat.ENCODING_PCM_16BIT,audioData.length, AudioTrack.MODE_STATIC);Log.d(TAG, "Writing audio data...");this.audioTrack.write(audioData, 0, audioData.length);Log.d(TAG, "Starting playback");audioTrack.play();

3.2、AudioTrack MODE_STREAM 使用

package com.ubtechinc.cruzr.voice.pcm;import android.media.AudioFormat;
import android.media.AudioManager;
import android.media.AudioTrack;
import androidx.annotation.NonNull;
import com.ubtrobot.cruzr.core.log.ELog;
import okio.ByteString;import static android.media.AudioTrack.PLAYSTATE_PLAYING;public class AudioTrackManager {private static final String TAG = "AudioTrackManager";private AudioTrack mAudioTrack;private volatile static AudioTrackManager mInstance;private long bufferCount;/*** 音频流类型*/private static final int mStreamType = AudioManager.STREAM_MUSIC;/*** 指定采样率 (MediaRecoder 的采样率通常是8000Hz AAC的通常是44100Hz。* 设置采样率为44100,目前为常用的采样率,官方文档表示这个值可以兼容所有的设置)*/private static final int mSampleRateInHz = 16000;/*** 指定捕获音频的声道数目。在AudioFormat类中指定用于此的常量*/private static final int mChannelConfig = AudioFormat.CHANNEL_CONFIGURATION_MONO; //单声道/*** 指定音频量化位数 ,在AudioFormaat类中指定了以下各种可能的常量。* 通常我们选择ENCODING_PCM_16BIT和ENCODING_PCM_8BIT PCM代表的是脉冲编码调制,它实际上是原始音频样本。* 因此可以设置每个样本的分辨率为16位或者8位,16位将占用更多的空间和处理能力,表示的音频也更加接近真实。*/private static final int mAudioFormat = AudioFormat.ENCODING_PCM_16BIT;/*** 指定缓冲区大小。调用AudioTrack类的getMinBufferSize方法可以获得。*/private int mMinBufferSize;/*** STREAM的意思是由用户在应用程序通过write方式把数据一次一次得写到audiotrack中。* 这个和我们在socket中发送数据一样,* 应用层从某个地方获取数据,例如通过编解码得到PCM数据,然后write到audiotrack。*/private static int mMode = AudioTrack.MODE_STREAM;private IAudioPlayStateListener iAudioPlayStateListener;private static final int BUFFER_CAPITAL = 10;/*** 获取单例引用** @return*/public static AudioTrackManager getInstance() {if (mInstance == null) {synchronized (AudioTrackManager.class) {if (mInstance == null) {mInstance = new AudioTrackManager();}}}return mInstance;}public AudioTrackManager() {initAudioTrack();}private void initAudioTrack() {//根据采样率,采样精度,单双声道来得到frame的大小。//计算最小缓冲区 *10mMinBufferSize = AudioTrack.getMinBufferSize(mSampleRateInHz, mChannelConfig, mAudioFormat);ELog.i(TAG, "initAudioTrack:  mMinBufferSize: " + mMinBufferSize * BUFFER_CAPITAL + " b");//注意,按照数字音频的知识,这个算出来的是一秒钟buffer的大小。mAudioTrack = new AudioTrack(mStreamType, mSampleRateInHz, mChannelConfig,mAudioFormat, mMinBufferSize * BUFFER_CAPITAL, mMode);}public void addAudioPlayStateListener(IAudioPlayStateListener iAudioPlayStateListener) {this.iAudioPlayStateListener = iAudioPlayStateListener;}public void prepareAudioTrack() {bufferCount = 0;ELog.i(TAG, "prepareAudioTrack:------> ");if (null == mAudioTrack) {return;}if (mAudioTrack.getState() == mAudioTrack.STATE_UNINITIALIZED) {initAudioTrack();}mAudioTrack.play();if (null != iAudioPlayStateListener) {iAudioPlayStateListener.onStart();}}public synchronized void write(@NonNull final ByteString bytes) {if (null != mAudioTrack) {int byteSize = bytes.size();bufferCount += byteSize;int write = mAudioTrack.write(bytes.toByteArray(), 0, bytes.size());ELog.d(TAG, "write: 接收到数据 " + byteSize + " b | 已写入 " + bufferCount + " b");if (write == 0 && null != iAudioPlayStateListener) {//由于缓存的缘故,会先把缓存的bytes填满再播放,当write=0的时候存在没有播完的情况try {Thread.sleep(2000);} catch (InterruptedException e) {e.printStackTrace();}if(iAudioPlayStateListener!=null){iAudioPlayStateListener.onStop();}}}}public void stopPlay() {ELog.i(TAG, "stopPlay: ");if (null == mAudioTrack) {return;}if (null != iAudioPlayStateListener) {iAudioPlayStateListener.onStop();}try {if (mAudioTrack.getPlayState() == PLAYSTATE_PLAYING) {mAudioTrack.stop();}} catch (IllegalStateException e) {ELog.e(TAG, "stop: " + e.toString());e.printStackTrace();}}public void release() {if (null == mAudioTrack) {return;}ELog.i(TAG, "release: ");stopPlay();iAudioPlayStateListener = null;try {mAudioTrack.release();mAudioTrack = null;} catch (Exception e) {ELog.e(TAG, "release: " + e.toString());e.printStackTrace();}}public void setBufferParams(int pcmFileSize) {//设置缓冲的大小 为PCM文件大小的10%ELog.d(TAG, "setFileSize: PCM文件大小为:" + pcmFileSize + " b 最小缓存空间为 " + mMinBufferSize * BUFFER_CAPITAL + " b");if (pcmFileSize < mMinBufferSize * BUFFER_CAPITAL) {mAudioTrack = new AudioTrack(mStreamType, mSampleRateInHz, mChannelConfig,mAudioFormat, mMinBufferSize, mMode);ELog.d(TAG, "setFileSize: pcmFileSize 文件小于最小缓冲数据的10倍,修改为默认的1倍------>");} else {//缓存大小为PCM文件大小的10%,如果小于mMinBufferSize * BUFFER_CAPITAL,则按默认值设置int cacheFileSize = (int) (pcmFileSize * 0.1);int realBufferSize = (cacheFileSize / mMinBufferSize + 1) * mMinBufferSize;ELog.d(TAG,"计算得到缓存空间为: "+realBufferSize+" b 最小缓存空间为 " + mMinBufferSize * BUFFER_CAPITAL + " b");if (realBufferSize < mMinBufferSize * BUFFER_CAPITAL) {realBufferSize=mMinBufferSize * BUFFER_CAPITAL;}mAudioTrack = new AudioTrack(mStreamType, mSampleRateInHz, mChannelConfig,mAudioFormat, realBufferSize, mMode);ELog.d(TAG, "setFileSize: 重置缓存空间为: " + realBufferSize + " b | "+realBufferSize/1024+" kb");}bufferCount = 0;}}

四、AudioTrack 和 MediaPlayer的对比

4.1 区别

  • 其中最大的区别是MediaPlayer可以播放多种格式的声音文件,例如MP3,AAC,WAV,OGG,MIDI等。MediaPlayer会在framework层创建对应的音频解码器。而AudioTrack只能播放已经解码的PCM流,如果对比支持的文件格式的话则是AudioTrack只支持wav格式的音频文件,因为wav格式的音频文件大部分都是PCM流。AudioTrack不创建解码器,所以只能播放不需要解码的wav文件。

4.2 联系

  • MediaPlayer在framework层还是会创建AudioTrack,把解码后的PCM数流传递给AudioTrack,AudioTrack再传递给AudioFlinger进行混音,然后才传递给硬件播放,所以是MediaPlayer包含了AudioTrack。

4.3 SoundPool

  • 在接触Android音频播放API的时候,发现SoundPool也可以用于播放音频。下面是三者的使用场景:MediaPlayer 更加适合在后台长时间播放本地音乐文件或者在线的流式资源; SoundPool 则适合播放比较短的音频片段,比如游戏声音、按键声、铃声片段等等,它可以同时播放多个音频; 而 AudioTrack 则更接近底层,提供了非常强大的控制能力,支持低延迟播放,适合流媒体和VoIP语音电话等场景。

使用AudioTrack播放pcm流式音频相关推荐

  1. Android编程中利用AudioTrack播放PCM数据在音频的最后出现重复回声现象的解决方案

    1 问题描述 今天在进行Android编程时遇到一个很奇怪的问题  重点说一下 这里我的测试机用的是"小米Note"  的确不怎么样 我在"华为P8"上面测试就 ...

  2. AudioTrack播放PCM音频

    一.前言 说到在 Android 平台上播放音频,我们最先想到的是 MediaPlayer.系统 API 对其做了比较全面的封装,开发者用少量的代码就能实现播放功能.MediaPlayer 可以播放多 ...

  3. AudioTrack播放pcm格式音频

    AudioTrack播放pcm格式音频 package com.zero.demo;import android.content.Context; import android.media.Audio ...

  4. NDK学习笔记:JNI调用Java层方法创建Native的AudioTrack播放PCM(方法签名,CallXXXMethod)

    NDK学习笔记:JNI调用Java层方法创建Native的AudioTrack播放PCM 题目有点复杂,不过确实就是那么回事.这章想记录的内容比较多,先列出来: native static 与 nat ...

  5. Android多媒体之SoundPool+pcm流的音频操作

    零.前言 今天比较简单,先理一下录制和播放的四位大将 再说一下SoundPool的使用和pcm转wav 讲一下C++文件如何在Android中使用,也就是传说中的JNI 最后讲一下变速播放和变调播放 ...

  6. AudioTrack 播放PCM音频数据

    AudioTrack 可以用来播放PCM数据,上一篇博客我讲了AudioRecord可以录制PCM数据 AudioTrack实例可以在两种模式下运行:静态或流式传输. 在Streaming模式下,应用 ...

  7. 音视频开发系列(28)AudioTrack播放PCM音频

    目录 AudioTrack和MediaPlayer AudioTrack的API介绍(构造.操作.状态机) 具体实现(Static和Stream两种模式) 遇到的问题 收获 一.MediaPlayer ...

  8. 51、流式音频之二(应用层)

    1.流式存储媒体 把注意力转移到网络应用,第一种情况针对早已存储在文件中的流媒体.最常见的例子是在Internet 上观看视频.这是视频点播( VoD, Video on Demand)的一种形式.其 ...

  9. Android AudioTrack播放PCM文件

    上篇文章写了使用AudioRecord采集音频,为了测试采集音频是否正确,可以通过AudioTrack播放音频试下. AudioTrack只能播放PCM格式的文件.PCM全称是Pulse Code M ...

最新文章

  1. 数据结构与算法(7-1)图的存储(邻接矩阵、邻接表)
  2. 用CSS3制作很特别的波浪形菜单
  3. 整数数组中最大子数组求和02
  4. C — 对C语言的认识
  5. mysql经常问到的面试题_20道BAT面试官最喜欢问的JVM+MySQL面试题(含答案解析)...
  6. kotlin学习之类委托(八)
  7. r语言模型评估:_情感分析评估:对自然语言处理的过去和未来的反思
  8. 用户id可以出现在url中吗_下载Google Drive中的文件
  9. 现代软件工程 第六章 【敏捷流程】练习与讨论
  10. 用按钮控制歌单的上一曲和下一曲 0130 winform
  11. MVC判断用是否登录了平台
  12. python开发mes系统_MES系统开发
  13. 集合的洗牌,排序,拆分以及常用遍历方法
  14. LinearLayout removeAllViews后再 addView页面不展示
  15. 微信java版s40_塞班微信S40版下载
  16. Java工程师学习指南(完结篇)
  17. GPS测量的精度及用途
  18. 前端处理后端返回的二进制流文件
  19. Unity学习笔记1 锁定摄像机镜头跟随角色移动
  20. 《点满舔狗相关技能以后,不知为何就转码了》之YOLOP,YOLOPV2推理入门教学

热门文章

  1. freepbx 脚本安装方法
  2. AB液体室温发泡硅胶(双组份室温发泡硅胶)
  3. 1^3 +2^3+3^3+...+n^3
  4. 普通电脑安装华为电脑管家操作流程
  5. Response Request 对象
  6. 健身耳机哪个牌子好,分享几款健身好用的耳机品牌
  7. 四款超实用的电脑软件,90%的人没有安装!
  8. Verilog语法规则
  9. Word通配符(正则表达式)批量替换举例
  10. 3-2 Verilog 4位行波进位加法器