本站文章均為?李華明Himi?原創,轉載務必在明顯處注明:
轉載自【黑米GameDev街區】?原文鏈接:?http://www.himigame.com/android-game/312.html
??? 游戲開發中,通過資料和書籍了解到在有兩種播放音頻形式可以用在我們的游戲開發中,第一個:MediaPlayer 類 ;第二個:SoundPool 類!
??? PS:當然還有一個JetPlayer 但是 播放的文件格式比較麻煩,所以這里拋開不解釋,有興趣的可以去自己研究下、呵呵;?
運行效果圖:?
?
??? MediaPlayer 和:SoundPool 類!那么他們之間的利弊各是什么呢?或者說,我們游戲開發到底用哪一個更佳呢?
??? 答案就是:兩者都必須要!!!分析利弊與各自的用途后,等各位童鞋熟習每個播放形式實現之后我會詳細道來!?
??? 下面仍然是先上代碼:(先看代碼 然后我講解兩個播放形式的利弊關系和各個用途以及其中解釋代碼中的幾個備注!)?
package?com.himi; ?import?java.util.HashMap; ?import?android.content.Context; ?import?android.graphics.Canvas; ?import?android.graphics.Color; ?import?android.graphics.Paint; ?import?android.media.AudioManager; ?import?android.media.MediaPlayer; ?import?android.media.SoundPool; ?import?android.view.KeyEvent; ?import?android.view.MotionEvent; ?import?android.view.SurfaceHolder; ?import?android.view.SurfaceView; ?import?android.view.SurfaceHolder.Callback; ?public?class?MySurfaceView?extends?SurfaceView?implements?Callback,?Runnable?{ ?????private?Thread?th; ?????private?SurfaceHolder?sfh; ?????private?Canvas?canvas; ?????private?MediaPlayer?player; ?????private?Paint?paint; ?????private?boolean?ON?=?true; ?????private?int?currentVol,?maxVol; ?????private?AudioManager?am;? ?????private?HashMap<Integer,?Integer>?soundPoolMap;//備注1 ?????private?int?loadId; ?????private?SoundPool?soundPool; ?????public?MySurfaceView(Context?context)?{ ?????????super(context); ?//?獲取音頻服務然后強轉成一個音頻管理器,后面方便用來控制音量大小用 ?????????am?=?(AudioManager)?MainActivity.instance ?????????????????.getSystemService(Context.AUDIO_SERVICE); ?????????maxVol?=?am.getStreamMaxVolume(AudioManager.STREAM_MUSIC); ?????????//?獲取最大音量值(15最大!?.不是100!) ?????????sfh?=?this.getHolder(); ?????????sfh.addCallback(this); ?????????th?=?new?Thread(this); ?????????this.setKeepScreenOn(true); ?????????setFocusable(true); ?????????paint?=?new?Paint(); ?????????paint.setAntiAlias(true); ?????????//MediaPlayer的初始化 ?????????player?=?MediaPlayer.create(context,?R.raw.himi);? ?????????player.setLooping(true);//設置循環播放 ?????????//SoundPool的初始化 ?????????soundPool?=?new?SoundPool(4,?AudioManager.STREAM_MUSIC,?100); ?????????soundPoolMap?=?new?HashMap<Integer,?Integer>(); ?????????soundPoolMap.put(1,?soundPool.load(MainActivity.content, ?????????????????R.raw.himi_ogg,?1)); ?????????loadId?=?soundPool.load(context,?R.raw.himi_ogg,?1); ?//load()方法的最后一個參數他標識優先考慮的聲音。目前沒有任何效果。使用了也只是對未來的兼容性價值。 ?????} ?????public?void?surfaceCreated(SurfaceHolder?holder)?{ ?????????/* ??????????*?Android?OS中,如果你去按手機上的調節音量的按鈕,會分兩種情況, ??????????*?一種是調整手機本身的鈴聲音量,一種是調整游戲,軟件,音樂播放的音量 ??????????*?當我們在游戲中的時候?,總是想調整游戲的音量而不是手機的鈴聲音量, ??????????*?可是煩人的問題又來了,我在開發中發現,只有游戲中有聲音在播放的時候 ??????????*?,你才能去調整游戲的音量,否則就是手機的音量,有沒有辦法讓手機只要是 ??????????*?在運行游戲的狀態就只調整游戲的音量呢?試試下面這段代碼吧! ??????????*/ ?????????MainActivity.instance.setVolumeControlStream(AudioManager.STREAM_MUSIC); ?????????//?設定調整音量為媒體音量,當暫停播放的時候調整音量就不會再默認調整鈴聲音量了,娃哈哈 ????????? ?????????player.start(); ?????????th.start(); ?????} ?????public?void?draw()?{ ?????????canvas?=?sfh.lockCanvas(); ?????????canvas.drawColor(Color.WHITE); ?????????paint.setColor(Color.RED); ?????????canvas.drawText("當前音量:?"?+?currentVol,?100,?40,?paint); ?????????canvas.drawText("當前播放的時間"?+?player.getCurrentPosition()?+?"毫秒",?100, ?????????????????70,?paint); ?????????canvas.drawText("方向鍵中間按鈕切換?暫停/開始",?100,?100,?paint); ?????????canvas.drawText("方向鍵←鍵快退5秒?",?100,?130,?paint); ?????????canvas.drawText("方向鍵→鍵快進5秒?",?100,?160,?paint); ?????????canvas.drawText("方向鍵↑鍵增加音量?",?100,?190,?paint); ?????????canvas.drawText("方向鍵↓鍵減小音量",?100,?220,?paint); ?????????sfh.unlockCanvasAndPost(canvas); ?????} ?????private?void?logic()?{ ?????????currentVol?=?am.getStreamVolume(AudioManager.STREAM_MUSIC);//?不斷獲取當前的音量值 ?????} ?????@Override ?????public?boolean?onKeyDown(int?key,?KeyEvent?event)?{ ?????????if?(key?==?KeyEvent.KEYCODE_DPAD_CENTER)?{ ?????????????ON?=?!ON; ?????????????if?(ON?==?false) ?????????????????player.pause(); ?????????????else ?????????????????player.start();? ?????????}?else?if?(key?==?KeyEvent.KEYCODE_DPAD_UP)?{//?按鍵這里本應該是RIGHT,但是因為當前是橫屏模式,以下雷同 ?????????????player.seekTo(player.getCurrentPosition()?+?5000); ?????????}?else?if?(key?==?KeyEvent.KEYCODE_DPAD_DOWN)?{ ?????????????if?(player.getCurrentPosition()?<?5000)?{ ?????????????????player.seekTo(0); ?????????????}?else?{ ?????????????????player.seekTo(player.getCurrentPosition()?-?5000); ?????????????} ?????????}?else?if?(key?==?KeyEvent.KEYCODE_DPAD_LEFT)?{ ?????????????currentVol?+=?1; ?????????????if?(currentVol?>?maxVol)?{ ?????????????????currentVol?=?100; ?????????????} ?????????????am.setStreamVolume(AudioManager.STREAM_MUSIC,?currentVol,//?備注2 ?????????????????????AudioManager.FLAG_PLAY_SOUND); ?????????}?else?if?(key?==?KeyEvent.KEYCODE_DPAD_RIGHT)?{ ?????????????currentVol?-=?1; ?????????????if?(currentVol?<=?0)?{ ?????????????????currentVol?=?0; ?????????????} ?????????????am.setStreamVolume(AudioManager.STREAM_MUSIC,?currentVol, ?????????????????????AudioManager.FLAG_PLAY_SOUND); ?????????} ?????????soundPool.play(loadId,?currentVol,?currentVol,?1,?0,?1f);//?備注3 ?//??????soundPool.play(soundPoolMap.get(1),?currentVol,?currentVol,?1,?0,?1f);//備注4 ?//??????soundPool.pause(1);//暫停SoundPool的聲音? ?????????return?super.onKeyDown(key,?event); ?????}? ?????@Override ?????public?boolean?onTouchEvent(MotionEvent?event)?{ ?????????return?true; ?????}? ?????public?void?run()?{ ?????????//?TODO?Auto-generated?method?stub ?????????while?(true)?{ ?????????????draw(); ?????????????logic(); ?????????????try?{ ?????????????????Thread.sleep(100); ?????????????}?catch?(Exception?ex)?{ ?????????????} ?????????} ?????}? ?????public?void?surfaceChanged(SurfaceHolder?holder,?int?format,?int?width, ?????????????int?height)?{?? ?????}? ?????public?void?surfaceDestroyed(SurfaceHolder?holder)?{?? ?????}? ?} ? 一、 MediaPlayer 播放音頻的實現步驟:
??? 1. 調用MediaPlayer.create(context, R.raw.himi); 利用MediaPlayer類調用create方法并且傳入通過id索引的資源音頻文件,得到實例;
??? 2. 得到的實例就可以調用 MediaPlayer.star();
簡單吧、其實MediaPlayer還有幾個構造方法,大家有興趣可以去嘗試和實現,這里主要是簡單的向大家介紹基本的,畢竟簡單實用最好!
?
二、 SoundPlayer 播放音頻的實現步驟:
??? 1. new出一個實例 ; new SoundPool(4, AudioManager.STREAM_MUSIC, 100);第一個參數是允許有多少個聲音流同時播放,第2個參數是聲音類型,第三個參數是聲音的品質;
??? 2.loadId = soundPool.load(context, R.raw.himi_ogg, 1);
??? 3. 使用實例調用play方法傳入對應的音頻文件id即可!?
下面講下兩個播放形式的利弊:?
??? 使用MediaPlayer來播放音頻文件存在一些不足:
??? 例如:資源占用量較高、延遲時間較長、不支持多個音頻同時播放等。
??? 這些缺點決定了MediaPlayer在某些場合的使用情況不會很理想,例如在對時間精準度要求相對較高的游戲開發中。
??? 最開始我使用的也是普通的MediaPlayer的方式,但這個方法不適合用于游戲開發,因為游戲里面同時播放多個音效是常有的事,用過MediaPlayer的朋友都該知道,它是不支持實時播放多個聲音的,會出現或多或少的延遲,而且這個延遲是無法讓人忍受的,尤其是在快速連續播放聲音(比如連續猛點按鈕)時,會非常明顯,長的時候會出現3~5秒的延遲,【使用MediaPlayer.seekTo() 這個方法來解決此問題】;
?
??? 相對于使用SoundPool存在的一些問題:
??? 1. SoundPool最大只能申請1M的內存空間,這就意味著我們只能使用一些很短的聲音片段,而不是用它來播放歌曲或者游戲背景音樂(背景音樂可以考慮使用JetPlayer來播放)。
??? 2. SoundPool提供了pause和stop方法,但這些方法建議最好不要輕易使用,因為有些時候它們可能會使你的程序莫名其妙的終止。還有些朋友反映它們不會立即中止播放聲音,而是把緩沖區里的數據播放完才會停下來,也許會多播放一秒鐘。
?
??? 3. 音頻格式建議使用OGG格式。使用WAV格式的音頻文件存放游戲音效,經過反復測試,在音效播放間隔較短的情況下會出現異常關閉的情況(有說法是SoundPool目前只對16bit的WAV文件有較好的支持)。后來將文件轉成OGG格式,問題得到了解決。
??? 4.在使用SoundPool播放音頻的時候,如果在初始化中就調用播放函數進行播放音樂那么根本沒有聲音,不是因為沒有執行,而是SoundPool需要一準備時間!囧。當然這個準備時間也很短,不會影響使用,只是程序一運行就播放會沒有聲音罷了,所以我把SoundPool播放寫在了按鍵中處理了、備注4的地方
?
大概看完了利弊解釋,那么來看我的代碼備注的地方:
備注1:
??? 這里我定義了一個 HashMap ,這個是哈希表,如果大家不是很了解這個類,那建議百度 google學習下,它與Hashtable很常用的,它倆的主要區別是: HashMap 不同步、空鍵值、效率高; Hashtable 同步、非空鍵值、效率略低 ;而在J2ME中不支持HashMap ,因為me中不支持空鍵值,所以在me中只能使用hashtable、咳咳、言歸正傳,我這里使用hashmap主要是為了存入多個音頻的ID,播放的時候可以同時播放多個音頻。
??? 上面也介紹了,SoundPool可以支持多個音頻同時播放,而且SoundPool在播放的時候調用的這個方法(備注3)soundPool.play(loadId, currentVol, currentVol, 1, 0, 1f); 第一個參數指的就是之前的loadId !是通過 soundPool.load(context, R.raw.himi_ogg, 1);方法取出來的,
??? 那么除此之外還要注意一點的就是定義hashmap的時候一定要定義成這種形式HashMap<Integer, Integer> hm = new Hash<Integer, Integer>,聲明此哈希表就是一個key和volue值都是Integer的哈希表! 為什么要這么做,因為如果你只是簡單的定義成 HashMap hm =new HashMap(),那么當你在播放的時候,也就是備注4方法這里的第一個id參數使用Hashmap.get()這個方法的時候總會出現錯誤的提示!?
??? 《SoundPool最大只能申請1M的內存空間,這就意味著我們只能使用一些很短的聲音片段》為什么只能使用一些很短的聲音呢?
??? 大家還是看備注4方法的第一個參數,這里要求傳入的Id類型是個int值,那么這個int其實對應的是通過load()方法返回的音頻id,而且這個id會因音頻文件的大小而變大變小,那么一旦我們的音頻文件超過int最大值,那么就會報內存錯誤的異常。所以為什么用SoundPool只能播放一些簡短的音頻這就是其原因了。當然os 里為什么這么定義 我也無從查證和說明。
?備注4 :此方法中參數的解釋
??? 第一個參數是我通過SoundPool.load()方法返回的音頻對應id,第二個第三個參數表示左右聲道大小,第四個參數是優先級,第五個參數是循環次數,最后一個是播放速率(1.0 =正常播放,范圍是0.5至2.0)
?
備注2:
??? 這里是通過媒體服務得到一個音頻管理器,從而來對音量大小進行調整。這里要強調一下,調整音頻是用這個音頻管理器調用setStreamVolume()的方式去調整,而不是MediaPlayer.setVolue(int LeftVolume,int RightVolume);這個方法的兩個參數也是調正左右聲道而不是調節聲音大小。?
??? 好了,對此我們對游戲開發中到底需要用什么來做進行了分析,總結就是SoundPool適合做特效聲,其實播放背景音樂我感覺還是用MediaPlayer比較好,當然啦,用什么都看大家喜好和選擇啦!下面附上項目下載地址:(項目10+MB因為含有res音頻文件)?
有人問 怎么才知道一首歌曲播放完了,那么這里給說下:?
??? PlaybackCompleted狀態:文件正常播放完畢,而又沒有設置循環播放的話就進入該狀態,并會觸發OnCompletionListener的onCompletion()方法。此時可以調用start()方法重新從頭播放文件,也可以stop()停止MediaPlayer,或者也可以seekTo()來重新定位播放位置。
注意:1、 別忘記綁定操作! mp.setOnCompletionListener(this);
???????2、如果你設置了循環播放 mp.setLooping(true); 的話,那么永遠都不會監聽到播放完成的狀態!!!!這里一定要注意!
??×××地址:?http://www.himigame.com/android-game/312.html
(推薦大家訂閱本博客,因為咱的更新速度可是很快的~娃哈哈)
轉載于:https://blog.51cto.com/xiaominghimi/606131
總結
以上是生活随笔為你收集整理的【Android游戏开发之八】游戏中添加音频-详解MediaPlayer与SoundPool的利弊以及各个在游戏中的用途!...的全部內容,希望文章能夠幫你解決所遇到的問題。
如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。