为什么80%的码农都做不了架构师?>>>   

 李华明Himi 原创,转载务必在明显处注明:
转载自 【黑米GameDev街区】 原文链接:  http://www.himigame.com/android-game/312.html

很多童鞋说我的代码运行后,点击home或者back后会程序异常,如果你也这样遇到过,那么你肯定没有仔细读完Himi的博文,第十九篇Himi专门写了关于这些错误的原因和解决方法,这里我在博客都补充说明下,省的童鞋们总疑惑这一块;请点击下面联系进入阅读:

【Android游戏开发十九】(必看篇)SurfaceView运行机制详解—剖析Back与Home按键及切入后台等异常处理!

游戏开发中,通过资料和书籍了解到在有两种播放音频形式可以用在我们的游戏开发中,第一个: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;//备注1private 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);// 不断获取当前的音量值}@Overridepublic boolean onKeyDown(int key, KeyEvent event) {if (key == KeyEvent.KEYCODE_DPAD_CENTER) {ON = !ON;if (ON == false)player.pause();elseplayer.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,// 备注2AudioManager.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);} @Overridepublic boolean onTouchEvent(MotionEvent event) {return true;} public void run() {// TODO Auto-generated method stubwhile (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

(推荐大家订阅本博客,因为咱的更新速度可是很快的~娃哈哈)

原文链接: http://blog.csdn.net/xiaominghimi/article/details/6101737

转载于:https://my.oschina.net/liux/blog/49868

【Android游戏开发之八】游戏中添加音频-详解MediaPlayer与SoundPoo!并讲解两者的区别和游戏中的用途!...相关推荐

  1. 【Android游戏开发之八】游戏中添加音频-详解MediaPlayer与SoundPool的利弊以及各个在游戏中的用途!...

    本站文章均为 李华明Himi 原创,转载务必在明显处注明: 转载自[黑米GameDev街区] 原文链接: http://www.himigame.com/android-game/312.html 游 ...

  2. Android 应用开发(41)---EditText(输入框)详解

    EditText(输入框)详解 1.设置默认提示文本 如下图,相信你对于这种用户登录的界面并不陌生,是吧,我们很多时候都用的这种界面 相比另外这种,下面这种又如何? 还不赖是吧,当然,不会在这里贴布局 ...

  3. linux中groupadd命令详解,adduser和addgroup命令讲解

    在Alpine Linux下创建用户和组命令分别是adduser和addgroup,本文主要介绍这两个命令相关参数及使用. adduser adduser [OPTIONS] USER [GROUP] ...

  4. java字符编码详解_Java中字符编码格式详解

    一.前言 在分析Comparable和Comparator的时候,分析到了String类的compareTo方法,String底层是用char[]数组来存放元素,在比较的时候是比较的两个字符串的字符, ...

  5. linux pmap命令,linux中 pmap 命令详解

    通过查看帮助,返回了如下信息: Usage: pmap [options] pid [pid ...] Options: -x,--extended show details -X show even ...

  6. Java虚拟机中类加载机制详解

    Java虚拟机中类加载机制详解 1,什么是java类加载机制 **首先在java中,是通过编译来生成.class文件(可能在本地,或者网页下载),java的类加载机制就是 将这些.class文件加载到 ...

  7. 【iOS-Cocos2d游戏开发之十四】音频/音效/视频播放(利用Cocos2D-iPhone-Extensions嵌入Cocos2d进行视频播放!)...

    李华明Himi 原创,转载务必在明显处注明: 转载自[黑米GameDev街区] 原文链接: http://www.himigame.com/iphone-cocos2d/482.html 自从Himi ...

  8. protobuf前后端解析_前端后台以及游戏中使用google-protobuf详解

    前端后台以及游戏中使用google-protobuf详解 [TOC] 0.什么是protoBuf protoBuf是一种灵活高效的独立于语言平台的结构化数据表示方法,与XML相比,protoBuf更小 ...

  9. 饥荒服务器不显示管理员,饥荒联机版管理员怎么添加_饥荒联机版管理员介绍与添加方法详解_玩游戏网...

    <饥荒>联机版里面的管理员这个概念大家了解吗?我之前也不清楚管理员相关内容,下面笔者就为大家带来了饥荒联机版管理员介绍与添加方法详解,小伙伴们还不了解联机版管理员的下面跟我一起来看看吧. ...

最新文章

  1. Intel SGX Remote Attestation实例代码安装和执行,笔记
  2. 成都铁路警方联合多部门开展反恐防暴演练
  3. Leetcode0037--Sudoku Solver 数独游戏
  4. [摘抄] Bezier曲线、B样条和NURBS
  5. 【CV】使用直方图处理进行颜色校正
  6. Spring MVC集成Log4j
  7. 无盘服务器读写缓存,无盘系统中实现网络磁盘本地写缓存控制的方法
  8. RTX5 | 线程标志组02 - 获取线程标志与清除线程标志
  9. 方差-偏差平衡(Bias-Variance Balance)与模型选择
  10. 计算机二级java邓伦单_2010年3月计算机等级考试二级Java笔试试题(文字版)
  11. Git:使用 GitHub 托管代码的简单流程
  12. thinkphp去重,distinct、group
  13. 单例模式中的懒汉模式和饿汉模式
  14. android手机解除root,手机一键ROOT以后如何解除?手机root后怎么恢复
  15. html制作页面书签的步骤,网页制作基础教程(二)html标签
  16. 5福最多多少钱_2018支付宝集5福攻略介绍 一天内集齐不是没可能
  17. 如何获取微信开发者id及设置微信授权目录
  18. the little scheme Y-combinator
  19. Vision Transformer 必读系列之图像分类综述(二): Attention-based
  20. 为什么在校大学生都要考计算机证书?要考几个证?

热门文章

  1. linux redis客户端_10个 Linux 顶级开源缓存工具
  2. vba搜索java里面的sql_Excel VBA+SQL 代替Vlookup精确查找
  3. k8s停止服务_Kubernetes 服务部署最佳实践(二) 如何提高服务可用性
  4. android鸿洋布局,Android基础ConstrainLayout约束布局的介绍和使用
  5. 全民果园为什么服务器在维修中,你在果园管理中遇到的问题这里可以解决
  6. js中比较到期时间和当前时间
  7. 2018 ACM/ICPC 南京站小结
  8. vue 调用c++_electron-vue跨平台桌面应用开发实战教程(七)——ffi调用C++(macOS平台)...
  9. idea @value提示_IDEA 中springboot 项目使用 注解Autowired 出现红线
  10. 2017.9.1 最小生成树 失败总结