長音頻播放聲音分為MediaPlayer和AudioTrack兩種方案的。MediaPlayer可以播放多種格式的聲音文件,例如MP3,WAV,OGG,AAC,MIDI等。然而AudioTrack只能播放PCM數(shù)據(jù)流。當(dāng)然兩者之間還是有緊密的聯(lián)系,MediaPlayer在播放音頻時,在framework層還是會創(chuàng)建AudioTrack,把解碼后的PCM數(shù)流傳遞給AudioTrack,最后由AudioFlinger進(jìn)行混音,傳遞音頻給硬件播放出來。利用AudioTrack播放只是跳過Mediaplayer的解碼部分而已。
短音效播放聲音目前使用SoundPool方案,SoundPool 是 Android 提供的一個API類,用來播放簡短的音頻,使用簡單但功能相對強(qiáng)大。只需花很少的氣力,就可以完成音頻的播放、暫停、恢復(fù)及停止等操作。從名字上也可以看出,它是一個“pool”,我們可以加載多個音頻資源到內(nèi)存,進(jìn)行管理與播放,比如控制同時播放流的最大數(shù)目。加載資源到內(nèi)存是需要花費少許時間的,因此我們需要監(jiān)聽加載資源完畢的事件,在加載完畢后才能進(jìn)行播放,以免發(fā)生不可預(yù)期的錯誤。
以上均是GPU處理的硬編解碼方案。
1、MediaPlayer
使用注意點:
1、使用MediaPlayer的setDataSource設(shè)置raw資源需要轉(zhuǎn)化成Uri:
<pre>Uri setDataSourceuri = Uri.parse("android.resource://com.klzz.vipthink.pad/" + resId);</pre>
2、prepare()的調(diào)用可能需要很長時間執(zhí)行,因為它可能涉及到獲取和解碼媒體數(shù)據(jù)。因此,就像任何需要很長時間才能執(zhí)行的方法一樣,永遠(yuǎn)不要從應(yīng)用程序的UI線程調(diào)用它。這樣做會導(dǎo)致UI掛起,直到方法返回,這是一種非常糟糕的用戶體驗,并可能導(dǎo)致ANR(應(yīng)用程序沒有響應(yīng))錯誤。
準(zhǔn)備數(shù)據(jù)并且響應(yīng)UI的時間閾值是100ms。所以prepareAsync()可以解決以上問題。
3、可以通過調(diào)用start()、pause()和seekTo()等方法在start、pause()和PlaybackCompleted狀態(tài)之間切換。但是,調(diào)用stop()時,資源已經(jīng)釋放,此時不適合調(diào)用start(),需要重新加載數(shù)據(jù)。
2、AudioTrack
作用:AudioTrack是管理和播放單一音頻資源的類。AudioTrack僅僅能播放已經(jīng)解碼的PCM流,用于PCM音頻流的回放。
AudioTrack實現(xiàn)PCM音頻播放五步走
1、配置基本參數(shù)
StreamType音頻流類型,系統(tǒng)將幾種聲音的數(shù)據(jù)分開管理,STREAM參數(shù)對AudioTrack來說,它的含義就是告訴系統(tǒng),我現(xiàn)在想使用的是音樂播放還是系統(tǒng)播放還是電話鈴聲等等,方便系統(tǒng)對應(yīng)管理。
MODE模式(static和stream兩種)。stream一邊加載編解碼一邊播放,優(yōu)點可處理大音頻,缺點是效率低;static是一次性加載,缺點是不適用大音頻。
采樣率 (MediaRecoder 的采樣率通常是8000Hz AAC的通常是44100Hz。 設(shè)置采樣率為44100,目前為常用的采樣率,官方文檔表示這個值可以兼容所有的設(shè)置)
通道數(shù)目:最大為雙聲道。
頻量化位數(shù):mAudioFormat(只支持8bit和16bit兩種。)
2、獲取最小緩沖區(qū)大小
3、創(chuàng)建AudioTrack對象
4、獲取PCM數(shù)據(jù),轉(zhuǎn)成DataInputStream
- 第一種加載PCM的路徑獲取到PCM文件
- 第二種通過ffmpeg軟編碼轉(zhuǎn)碼其他類型的文件變?yōu)镻CM數(shù)據(jù)
- 第三種通過MediaCodec硬編碼轉(zhuǎn)化PCM數(shù)據(jù)
5、開啟/停止播放
3、SoundPool
作用:播放一些短時間且可以重復(fù)重疊的音效,是一個多軌道播放的音效音頻工具類。
注意事項:
1、pause()、resume()和stop()是針對播放流操作的,傳遞的是play()返回的streamID ;
2、play()中的priority參數(shù),只在同時播放的流的數(shù)量超過了預(yù)先設(shè)定的最大數(shù)量是起作用,管理器將自動終止優(yōu)先級低的播放流。如果存在多個同樣優(yōu)先級的流,再進(jìn)一步根據(jù)其創(chuàng)建事件來處理,新創(chuàng)建的流的年齡是最小的,將被終止;
3、同時播放多個音頻,通過play()函數(shù),成功則返回非0的streamID;
4、當(dāng)設(shè)置為無限循環(huán)時,需要手動調(diào)用stop()來終止播放;
5、播放流的優(yōu)先級(play()中的priority參數(shù)),只在同時播放數(shù)超過設(shè)定的最大數(shù)時起作用;
6、程序中不用考慮(play觸發(fā)的)播放流的生命周期,無效的soundID/streamID不會導(dǎo)致程序錯誤。
無論如何,程序退出時,手動終止播放并釋放資源是必要的。
7、soundID是使用在play方法上的并返回streamID,streamID是使用在stop方法上的,用于指定暫停指定音效。
開發(fā)中曾遇到過播放長語音(文件大小1M以上)不全,播放語音解碼損壞的問題。