java 开源 Ubuntu Android 程序员 google mysql Firefox 云计算 shell Python wordpress nginx php centos Windows apache 微软 编程 linux

【Android遊戲開發之八】遊戲中添加音頻-詳解MediaPlayer與SoundPool的利弊以及各個在遊戲中的用途!

 遊戲開發中,通過資料和書籍了解到在有兩種播放音頻形式可以用在我們的遊戲開發中,第一個: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);  
        // 設定調整音量為媒體音量,當暫停播放的時候調整音量就不會再默認調整鈴聲音量了,娃哈哈  

  1.           
  2.         player.start();  
  3.         th.start();  
  4.     }  
  5.     public void draw() {  
  6.         canvas = sfh.lockCanvas();  
  7.         canvas.drawColor(Color.WHITE);  
  8.         paint.setColor(Color.RED);  
  9.         canvas.drawText("当前音量: " + currentVol, 100, 40, paint);  
  10.         canvas.drawText("当前播放的时间" + player.getCurrentPosition() + "毫秒", 100,  
  11.                 70, paint);  
  12.         canvas.drawText("方向键中间按钮切换 暂停/开始", 100, 100, paint);  
  13.         canvas.drawText("方向键←键快退5秒 ", 100, 130, paint);  
  14.         canvas.drawText("方向键→键快进5秒 ", 100, 160, paint);  
  15.         canvas.drawText("方向键↑键增加音量 ", 100, 190, paint);  
  16.         canvas.drawText("方向键↓键减小音量", 100, 220, paint);  
  17.         sfh.unlockCanvasAndPost(canvas);  
  18.     }  
  19.     private void logic() {  
  20.         currentVol = am.getStreamVolume(AudioManager.STREAM_MUSIC);// 不断获取当前的音量值  
  21.     }  
  22.     @Override  
  23.     public boolean onKeyDown(int key, KeyEvent event) {  
  24.         if (key == KeyEvent.KEYCODE_DPAD_CENTER) {  
  25.             ON = !ON;  
  26.             if (ON == false)  
  27.                 player.pause();  
  28.             else  
  29.                 player.start();   
  30.         } else if (key == KeyEvent.KEYCODE_DPAD_UP) {// 按键这里本应该是RIGHT,但是因为当前是横屏模式,以下雷同  
  31.             player.seekTo(player.getCurrentPosition() + 5000);  
  32.         } else if (key == KeyEvent.KEYCODE_DPAD_DOWN) {  
  33.             if (player.getCurrentPosition() < 5000) {  
  34.                 player.seekTo(0);  
  35.             } else {  
  36.                 player.seekTo(player.getCurrentPosition() - 5000);  
  37.             }  
  38.         } else if (key == KeyEvent.KEYCODE_DPAD_LEFT) {  
  39.             currentVol += 1;  
  40.             if (currentVol > maxVol) {  
  41.                 currentVol = 100;  
  42.             }  
  43.             am.setStreamVolume(AudioManager.STREAM_MUSIC, currentVol,// 备注2  
  44.                     AudioManager.FLAG_PLAY_SOUND);  
  45.         } else if (key == KeyEvent.KEYCODE_DPAD_RIGHT) {  
  46.             currentVol -= 1;  
  47.             if (currentVol <= 0) {  
  48.                 currentVol = 0;  
  49.             }  
  50.             am.setStreamVolume(AudioManager.STREAM_MUSIC, currentVol,  
  51.                     AudioManager.FLAG_PLAY_SOUND);  
  52.         }  
  53.         soundPool.play(loadId, currentVol, currentVol, 1, 0, 1f);// 备注3  
  54. //      soundPool.play(soundPoolMap.get(1), currentVol, currentVol, 1, 0, 1f);//备注4  
  55. //      soundPool.pause(1);//暂停SoundPool的声音   
  56.         return super.onKeyDown(key, event);  
  57.     }   
  58.     @Override  
  59.     public boolean onTouchEvent(MotionEvent event) {  
  60.         return true;  
  61.     }   
  62.     public void run() {  
  63.         // TODO Auto-generated method stub  
  64.         while (true) {  
  65.             draw();  
  66.             logic();  
  67.             try {  
  68.                 Thread.sleep(100);  
  69.             } catch (Exception ex) {  
  70.             }  
  71.         }  
  72.     }   
  73.     public void surfaceChanged(SurfaceHolder holder, int format, int width,  
  74.             int height) {    
  75.     }   
  76.     public void surfaceDestroyed(SurfaceHolder holder) {    
  77.     }   
  78. }  

一、 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); 的話,那麽永遠都不會監聽到播放完成的狀態!!!!這裏一定要註意!

延伸阅读

评论