Android--SoundPool

前言

  在Android中播放音频文件经常会用到MediaPlayer,但是MediaPlayer存在一些不足的地方,如:资源占用量较高、加载延迟时间较长、不支持多个音频同时播放等。这些缺点决定了MediaPlayer在某些需要密集使用不同音频的情况不会理想,例如游戏开发。在游戏开发中,我们经常需要播放一些游戏的音效,这些音效的都需要是短促、密集、延迟小的,在这种场景下,需要使用到SoundPool来替代MediaPlayer播放这些音效,本篇博客就主要讲解SoundPool的使用以及需要注意的地方,最后将以一个示例演示SoundPool的使用。

  本篇博客的主要内容:

  1. SoundPool
  2. SoundPool的简单示例
  3. SoundPool的注意事项 

  

SoundPool

  SoundPool(声音池),所处于"android.media.SoundPool"包下,主要用于播放一些较短的声音片段,支持从程序的资源或文件系统加载。与MediaPlayer相比,SoundPool的优势在于CPU的资源占用量低、反应延迟小,并且可以加载多个音频到SoundPool中,通过资源ID来管理。另外SoundPool还支持执行设置声音的品质、音量、播放比率等参数。

  SoundPool提供一个构造函数,以下是它的完整签名:

    SoundPool(int maxStreams,int streamType,int srcQuality)

  通过上面的构造函数即可完成SoundPool的初始化,第一个参数为音频池最多支持装载多少个音频,就是音频池的大小;第二个参数指定声音的类型,在AudioManager类中以常量的形式定义,一般指定为AudioManager.STREAM_MUSIC即可;第三个参数为音频的质量,默认为0,这个参数为预留参数,现在没有实际意义,为扩展预留字段,一般传0即可。

 

  对于一个音频池,涉及到音频的加载、播放、暂停、继续、释放资源等操作,SoundPool也为我们提供了相应的方法,其底层也是用C++编写的native方法。以下介绍一些常用的SoundPool方法:

  • int load(Context context,int resId,int priority):从一个文件夹raw下装载一段音频资源,返回值为音频资源在SoundPool的ID。
  • int load(String path,int priority):从一个资源文件的路径装载一段音频资源,返回值为音频资源在SoundPool的ID。
  • final int play(int soundID,float leftVolume,float rightVolume,int priority,int loop,float rate):根据资源ID,播放一段音频资源。
  • final void pause(int streamID):根据装载资源ID,暂停音频资源的播放。
  • final void resume(int streamID):根据装载资源ID,继续播放暂停的音频资源。
  • final void stop(int streamID):根据装载资源ID,停止音频资源的播放。
  • final boolean unload(int soundID) :从音频池中卸载音频资源ID为soundID的资源。
  • final void release():释放音频池资源。

  上面方法无疑Load()和play()是最重要的,Load()具有多种重载方法,从参数名就可以看出是什么意思。这里讲解一下play()方法,soundID参数为资源ID;leftVolume和rightVolume个参数为左右声道的音量,从大到小取0.0f~1.0f之间的值;priority为音频质量,暂时没有实际意义,传0即可;loop为循环次数,0为播放一次,-1为无线循环,其他正数+1为播放次数,如传递3,循环播放4次;rate为播放速率,从大到小取0.0f~2.0f,1.0f为正常速率播放。

  在使用load()装载音频的时候需要注意,load()方法是一个异步的方法,也就是说,在播放音频的时候,很可能此段音频还没有装载到音频池中,这里可以借助SoundPool的一个装载完成的监听事件SoundPool.setOnLoadCompleteListener来保证装载完成在播放声音。SoundPool.setOnLoadCompleteListener()需要实现一个SoundPool.OnLoadCompleteListener接口,其中需要实现onLoadComplete()方法,一下是onLoadComplete()方法的完整签名:

    onLoadComplete(SoundPool soundPool, int sampleId, int status)

  • soundPool:当前触发事件的声音池。
  • sampleId:当前装载完成的音频资源在音频池中的ID。
  • status:状态码,展示没有意义,为预留参数,会传递0。

 

使用SoundPool示例

  上面已经介绍了SoundPool的使用所涉及到的内容,下面通过一个简单的示例来演示一下SoundPool的使用,播放的音频资源都是我从其他app中拷贝出来的,没有实际意义。示例中的注释写的比较全,这里不再累述了。

 1 package cn.bgxt.soundpooldemo;
 2 
 3 import java.util.HashMap;
 4 import java.util.Map;
 5 
 6 import android.media.AudioManager;
 7 import android.media.SoundPool;
 8 import android.media.SoundPool.OnLoadCompleteListener;
 9 import android.os.Bundle;
10 import android.app.Activity;
11 import android.util.Log;
12 import android.view.View;
13 import android.view.View.OnClickListener;
14 import android.widget.Button;
15 import android.widget.Toast;
16 
17 public class MainActivity extends Activity {
18     private Button btn_newqqmsg, btn_newweibontf, btn_newweibotoast;
19     private SoundPool pool;
20     private Map<String, Integer> poolMap;
21 
22     @Override
23     protected void onCreate(Bundle savedInstanceState) {
24         super.onCreate(savedInstanceState);
25         setContentView(R.layout.activity_main);
26         btn_newqqmsg = (Button) findViewById(R.id.btn_newqqmsg);
27         btn_newweibontf = (Button) findViewById(R.id.btn_newweibontf);
28         btn_newweibotoast = (Button) findViewById(R.id.btn_newweibotoast);
29 
30         poolMap = new HashMap<String, Integer>();
31         // 实例化SoundPool,大小为3
32         pool = new SoundPool(3, AudioManager.STREAM_MUSIC, 0);
33         // 装载音频进音频池,并且把ID记录在Map中
34         poolMap.put("newqqmsg", pool.load(this, R.raw.qqmsg, 1));
35         poolMap.put("newweibontf", pool.load(this, R.raw.notificationsound, 1));
36         poolMap.put("newweibotoast", pool.load(this, R.raw.newblogtoast, 1));
37 
38         pool.setOnLoadCompleteListener(new OnLoadCompleteListener() {
39 
40             @Override
41             public void onLoadComplete(SoundPool soundPool, int sampleId,
42                     int status) {
43                 // 每次装载完成均会回调
44                 Log.i("main", "音频池资源id为:" + sampleId + "的资源装载完成");
45                 // 当前装载完成ID为map的最大值,即为最后一次装载完成
46                 if (sampleId == poolMap.size()) {
47                     Toast.makeText(MainActivity.this, "加载声音池完成!",
48                             Toast.LENGTH_SHORT).show();
49                     btn_newqqmsg.setOnClickListener(click);
50                     btn_newweibontf.setOnClickListener(click);
51                     btn_newweibotoast.setOnClickListener(click);
52                     // 进入应用播放四次声音
53                     pool.play(poolMap.get("newweibotoast"), 1.0f, 1.0f, 0, 3,
54                             1.0f);
55                 }
56             }
57         });
58     }
59 
60     private View.OnClickListener click = new OnClickListener() {
61 
62         @Override
63         public void onClick(View v) {
64 
65             switch (v.getId()) {
66             case R.id.btn_newqqmsg:
67                 if (pool != null) {
68                     pool.play(poolMap.get("newqqmsg"), 1.0f, 1.0f, 0, 0, 1.0f);
69                 }
70                 break;
71             case R.id.btn_newweibontf:
72                 if (pool != null) {
73                     pool.play(poolMap.get("newweibontf"), 1.0f, 1.0f, 0, 0,
74                             1.0f);
75                 }
76                 break;
77             case R.id.btn_newweibotoast:
78                 if (pool != null) {
79                     pool.play(poolMap.get("newweibotoast"), 1.0f, 1.0f, 0, 0,
80                             1.0f);
81                 }
82                 break;
83             default:
84                 break;
85             }
86         }
87     };
88 
89     @Override
90     protected void onDestroy() {
91         // 销毁的时候释放SoundPool资源
92         if (pool != null) {
93             pool.release();
94             pool = null;
95         }
96         super.onDestroy();
97     }
98 }

  效果展示:

 

SoundPool使用的注意事项

  因为SoundPool的一些设计上的BUG,从固件版本1.0开始就有些没有修复的,以后应该会慢慢修复。这里简单提一下:

  1. 虽然SoundPool可以装载多个音频资源,但是最大只能申请1MB的内存空间,这就意味着只能用使用它播放一些很短的声音片段,而不是用它来播放歌曲或者做游戏背景音乐。
  2. SoundPool提供的pause()、resume()、stop()最好不要轻易使用,因为它有时候会使程序莫名其妙的终止,如果使用,最好做大量的测试。而且有时候也不会立即终止播放声音,而是会等缓冲区的音频数据播放完才会停止。
  3. 虽然SoundPool比MediaPlayer的效率好,但也不是绝对不存在延迟的问题,尤其在那些性能不太好的手机中,SoundPool的延迟问题会更严重,但是现在一般的手机配置,那一点的延迟还是可以接受的。

 

  源码下载

 

总结

  本篇博客介绍了SoundPool的使用,虽然SoundPool还有一些不足的地方,当时对于应用中一些简短的特效音,如按键音、短消息音,或者一些游戏中密集的声音,如射击的枪声,爆破的声音等。都是可以使用SoundPool的,效率会比使用MediaPlayer要高的多。

 

 

posted @ 2013-11-25 08:38  承香墨影  阅读(9877)  评论(0编辑  收藏  举报