转:SoundPool的一些注意(可能会导致没声音)

SoundPool的用法就不再提了,网上资料多,就说说在实际应用中可能会遇到的一些奇葩的问题,这些问题应该是和底层实现上有关系。

1、AudioFlinger could not  create track, status: -12 

   SoundPool即音效池,在创建的时候 maxStream这个参数代表能够同时播放的最大音效数,这里切忌合理使用,写的太大后会报AudioFlinger could not  create track, status: -12 。。。。一旦报了这个错,你就听不到声音了,呵呵。

 2、256个音效

       当调用load方法的时候实际就是把音效加载到了 SoundPool中,此时返回的streamId其实就是该音效在SoundPool中的Id,这个ID从0还是1来着(有点记不清了) 递增,不过要注意的是,不要超过  256  这个临界点。也就是说第257个声音加载进去后,调用play方法其实是播不出来的,说不定还会挤掉一些前面加载好的声音。这个256的限制通过查看SDK源码基本就能了解清楚,它底层就那么实现的,用一个类似堆栈来存。

  3、unload方法和release方法

         如果你音效多,也不要指望unload方法来清除掉一些音效后再load新的进去,虽然unload后音效卸载了,但是前面分给它在SoundPool里面的Id可没有释放掉,也就是说这个时候你load新的进去只会在后面继续累加,然后累加多了就超过256了,然后就就听不到声音,然后就没有然后了。要想彻底清掉前面的音效请使用release方法,它会连内存中占用的资源一起释放掉。

   其他还有点什么呢,load需要一点点时间,load后不要马上unload,load ---play--unload的做法并不可取,不要load太大的音效,它只会申请1M的内存空间。SoundPool出错后通常会看到retuen的值是0。

    差不多了,打字费力啊。

 

原文地址:http://blog.sina.com.cn/s/blog_71d26ff00100uzci.html

下面的另一篇博客,也很不错,解决了我遇到的问题。

写小结的目的有两个: 其一,可以加深对所学的内容的理解与提高; 其二,分享一下我的心得,同时希望大家踊跃指正,共同学习。

经过这段时间的软件开发,对MediaPlayer与SoundPool的使用以及在开发过程中出现的问题做一下小结:

由于开始对Android的音频控制类陌生,所有在网上搜了部分资料进行了学习,由于网上的资料有点零散,零零碎碎的找到了可以设置android音频的类MediaPlayer,记得刚开始用的时候,照着网上的方法:

MediaPlayer mMediaPlayer = new MediaPlayer();

mMediaPlayer.setDataSource("/system/media/audio/ui/KeypressStandard.ogg");
      mMediaPlayer.prepare();
      mMediaPlayer.start();

如此写了,由于有异常发生,故改为:

try {
   MediaPlayer mMediaPlayer = new MediaPlayer();
mMediaPlayer.setDataSource("/system/media/audio/ui/camera_focus.ogg");
mMediaPlayer.prepare();
mMediaPlayer.start();

} catch (IllegalArgumentException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IllegalStateException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}

起初感觉效果还不错,可是多按了几下,并出现了错误信息logcat

06-27 14:32:53.048: E/AudioSink(118): Unable to create audio track
06-27 14:32:53.048: E/MediaPlayer(1854): error (-19, 0)
06-27 14:32:53.048: I/st(1854): cackle
06-27 14:32:53.058: I/st(1854): cackle
06-27 14:32:53.058: D/MediaPlayerService(118): MediaPlayerService::Client::setDataSource() : screen = 0
06-27 14:32:53.058: I/AwesomePlayer(118): setDataSource_l(‘/system/media/audio/ui/camera_focus.ogg‘)
06-27 14:32:53.088: E/AudioFlinger(118): no more track names available
06-27 14:32:53.088: E/AudioTrack(118): AudioFlinger could not create track, status: -12
06-27 14:32:53.088: E/AudioSink(118): Unable to create audio track
06-27 14:32:53.088: E/MediaPlayer(1854): error (-19, 0)
06-27 14:32:53.118: E/MediaPlayer(1854): Error (-19,0)
06-27 14:32:53.118: E/MediaPlayer(1854): Error (-19,0)

后来改用了SoundPool类,具体说下SoundPool类的使用方法:

1. 创建一个SoundPool

  public SoundPool(int maxStream, int streamType, int srcQuality)

  maxStream —— 同时播放的流的最大数量

  streamType —— 流的类型,一般为STREAM_MUSIC(具体在AudioManager类中列出)

  srcQuality —— 采样率转化质量,当前无效果,使用0作为默认值

  eg.

  SoundPool soundPool = new SoundPool(4, AudioManager.STREAM_MUSIC, 0);

  创建了一个最多支持3个流同时播放的,类型标记为音乐的SoundPool。

2 一般把多个声音放到HashMap中去

soundPool = new SoundPool(4, AudioManager.STREAM_MUSIC, 100); 
    soundPoolMap = new HashMap<Integer, Integer>();   
    soundPoolMap.put(1, soundPool.load(this, R.raw.dingdong, 1));

soundpool的加载: 
      int  load(Context context, int resId, int priority)  //从APK资源载入 
  int  load(FileDescriptor fd, long offset, long length, int priority)  //从FileDescriptor对象载入 
  int  load(AssetFileDescriptor afd, int priority)  //从Asset对象载入 
  int  load(String path, int priority)  //从完整文件路径名载入 
最后一个参数为优先级。

3 播放

play(int soundID, float leftVolume, float rightVolume, int priority, int loop, float rate) ,其中leftVolume和rightVolume表示左右音量,priority表示优先级,loop表示循环次数,rate表示速率,如 
//速率最低0.5最高为2,1代表正常速度
sp.play(soundId, 1, 1, 0, 0, 1); 
  而停止则可以使用 pause(int streamID) 方法,这里的streamID和soundID均在构造SoundPool类的第一个参数中指明了总数量,而id从0开始。

按照如上方法编写了以下代码替换MidiaPlayer类:

private HashMap<Integer, Integer> soundPoolMap;
    private int loadId;  
    private SoundPool soundPool;

soundPool = new SoundPool(3, AudioManager.STREAM_MUSIC, 100);

soundPoolMap = new HashMap<Integer, Integer>();  
        
        soundPoolMap.put(1, soundPool.load("/system/media/audio/ui/KeypressStandard.ogg", 1));  
        loadId = soundPool.load("/system/media/audio/ui/KeypressStandard.ogg", 1);

soundPool.play(loadId, 1, 1, 1, 0, 1f);

如此不再有错误信息出现了,下面我就来总结一下:

由于我使用的音频文件是系统的音频文件,占用内存本身比较小,另外开发的软件相对来说比较大,因此,在调用MediaPlayer的时候,内存空间不足,故出现了以上的错误信息。

下面是MediaPlayer和SoundPool类的对比特性:

1.soundpool可以播一些短的反应速度要求高的声音, 
比如游戏中的爆破声,而mediaplayer适合播放长点的。 
2. SoundPool载入音乐文件使用了独立的线程,不会阻塞UI主线程的操作。但是这里如果音效文件过大没有载入完成,我们调用play方法时可能产生严 重的后果,这里Android SDK提供了一个SoundPool.OnLoadCompleteListener类来帮助我们了解媒体文件是否载入完成,我们重载 onLoadComplete(SoundPool soundPool, int sampleId, int status) 方法即可获得。 
3. 从上面的onLoadComplete方法可以看出该类有很多参数,比如类似id,是的SoundPool在load时可以处理多个媒体一次初始化并放入内存中,这里效率比MediaPlayer高了很多。 
4. SoundPool类支持同时播放多个音效,这对于游戏来说是十分必要的,而MediaPlayer类是同步执行的只能一个文件一个文件的播放。

以上,作为我的一点浅薄之见,不好之处,恳请多多指正,谢谢!

PS:这里有一篇不错的博客:http://blog.csdn.net/xiaominghimi/article/details/6101737【Android游戏开发之八】游戏中添加音频-详解MediaPlayer与SoundPoo!并讲解两者的区别和游戏中的用途!)里面讲的比较详细,供大家学习与参考!

原文地址:http://blog.csdn.net/j67065/article/details/7699460

时间: 2024-10-19 20:25:47

转:SoundPool的一些注意(可能会导致没声音)的相关文章

container在运行的时候重启 docker 服务, 可能会导致 container无法启动

如果有container在运行的时候重启 docker 服务, 可能会导致 container无法启动, 错误信息类似于 Error response from daemon: Cannot start container zookeeper: Error getting container ddf1dd91bbf46dc648268327f8f7c6fffaf2f19cda5cf1d97fdc701016d4332c from driver devicemapper: Error mounti

对“demo!demo.Index+HookProc::Invoke”类型的已垃圾回收委托进行了回调。这可能会导致应用程序崩溃、损坏和数据丢失。向非托管代码传递委托时,托管应用程序必须让这些委托保持活

对"demo!demo.Index+HookProc::Invoke"类型的已垃圾回收委托进行了回调.这可能会导致应用程序崩溃.损坏和数据丢失.向非托管代码传递委托时,托管应用程序必须让这些委托保持活动状态,直到确信不会再次调用它们. 解救办法: //保持活动 避免 回调过程 被垃圾回收 GCHandle.Alloc(委托); 对"demo!demo.Index+HookProc::Invoke"类型的已垃圾回收委托进行了回调.这可能会导致应用程序崩溃.损坏和数据丢

servlet请求编码与响应编码问题(编码不一致可能会导致乱码)

html中的编码 <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">这里不设置成中文编码无法写中文. jsp中的编码 <%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> conte

如果两段内存重叠,用memcpy函数可能会导致行为未定义

如果两段内存重叠,用memcpy函数可能会导致行为未定义,改进: void* memmove(void* str1,const void* str2,size_t n) { char* pStr1= (char*) str1; const char* pStr2=(const char*)str2; if (pStr1 < pStr2 ) { for(size_t i=0;i!=n;++i) { *(pStr1++)=*(pStr2++); } } else { pStr1+=n-1; pStr

ef core code first 模式提示&quot;可能会导致循环或多重级联路径&quot;问题

执行命令 dotnet ef mirations add "xxxxxx" dotnet ef database update 报错 将 FOREIGN KEY 约束 'FK_SkuPropertyItem_Sku_SkuId' 引入表 'SkuPropertyItem' 可能会导致循环或多重级联路径.请指定 ON DELETE NO ACTION 或 ON UPDATE NO ACTION,或修改其他 FOREIGN KEY 约束.无法创建约束.请参阅前面的错误消息 修改项目定义的d

360或者金山毒霸可能会导致HP网络打印机驱动安装失败“数据无效”的解决办法

360或者金山毒霸可能会导致HP网络打印机驱动安装失败“数据无效”的解决办法     同事办公室的打印机是网线接口的那种网络打印机,不是直接连到电脑的那种,他电脑安装了360和金山毒霸,WIN10下安装网络打印机死活安装不上,提示“数据无效”.百度了下找到了解决办法:https://jingyan.baidu.com/article/a948d6513ef4200a2ccd2e50.html 右击桌面上我的电脑→管理→服务→把device install service和device setup

对“XXX::Invoke”类型的已垃圾回收委托进行了回调。这可能会导致应用程序崩溃、损坏和数据丢失。向非托管代码传递委托时,托管应用程序必须让这些委托保持活动状态,直到确信不会再次调用它们

托管调试助手“CallbackOnCollectedDelegate”在“D:\XXX\XXX.vshost.exe”中检测到问题. 其他信息: 对“XXX+HookProc::Invoke”类型的已垃圾回收委托进行了回调.这可能会导致应用程序崩溃.损坏和数据丢失.向非托管代码传递委托时,托管应用程序必须让这些委托保持活动状态,直到确信不会再次调用它们. 经过搜索资料,发现出问题的原因是我的程序里回调函数作用域的问题 (_mouseHookCallBack) 报错前代码: private voi

对“xxx”类型的已垃圾回收委托进行了回调。这可能会导致应用程序崩溃、损坏和数据丢失。向非托管代码传递委托时,托管应用程序必须让这些委托保持活动状态,直到确信不会再次调用它们。 错误解决一例。

最近在写一个海康的门禁的自动监控刷卡事件的程序. 因为用c#写的,大家都知道c#是垃圾自动回收的.海康提供的api是用c++写的,要将处理的回调代码委托给api .刚开始的时候很顺利,但当运行一段时间就会报以下错误: 对“xxx”类型的已垃圾回收委托进行了回调.这可能会导致应用程序崩溃.损坏和数据丢失.向非托管代码传递委托时,托管应用程序必须让这些委托保持活动状态,直到确信不会再次调用它们. 大致的原因是:c#把回调函数资源回收了,导致api收到事件的时候执行回调出错. 网上的解决方案是将回调方

ubuntu14.04下外放有声音,耳机没声音

真是使用linux每天都有新的发现啊,今天早上起来用电脑,想听几首歌,插上耳机后发现没声音,我还以为是耳机坏了,就把耳机插在了手机上,发现耳机有声音,很纳闷,我象是不是电脑接口有问题了,但是在进系统的时候耳机会响一下,所以应该也不是接口问题,接着检查了一下电脑的音频设置以及驱动,发现都合适,我有把电脑重新启动了一下,发现还是不行,很狗血的是我把电脑合了,然后又打开,很神奇的有声音了,自己也搞不懂为什么,我就想电脑合起来的状态是什么,打开电源管理,发现合上电脑盖是挂起状态,简单说一下ubuntu的