Android中播放声音
在Android系统中,有两种播放声音的方式,一种是通过MediaPlayer,另外一种是通过SoundPool。前者主要用于播放长时间的音乐,而后者用于播放小段小段的音效,像按键音这种,其优点是资源占用了小,同时能够载入多个声音片段,再根据需要选择播放。下面分别介绍这两种方式:
1、MediaPlayer
MediaPlayer有两种创建方式,方式一:
1
|
MediaPlayer mp = new MediaPlayer() |
通过这种方式创建MediaPlayer对象,必须在创建对象之后使用下面的语句:
1
2
|
mp.setDataSource( "filePath" ); mp.prepare(); |
然后就是调用start()方法。注意使用这种方法设置播放的音乐的时候需要使用的是路径,这样对于一般的应用使用起来就不是很方便,不能直接打包到apk中。
方式二:
1
|
mp = MediaPlayer.create( this , R.raw.music); |
通过这种方式创建MediaPlayer对象,就不需要setDataSource()和prepare()了,直接start()就好了。
当需要停止播放音乐的时候,使用下面的方式:
1
2
3
4
|
if (mp.isPlaying()){ mp.stop(); } mp.reset(); |
需要注意的是,在stop之前一定要确认mp正在播放,因为如果已经停止播放了,再次stop会引发错误的,原因就是MediaPlayer有一个严格的生命周期,在错误的时期调用错误的方法,是一定会报错的。
2、SoundPool
SoundPool在播放音效的时候特别的好用,为什么呢?因为它可以一次load较多的声音片段。下面首先介绍其基本的使用方法:
1
2
3
4
5
6
7
|
//创建 SoundPool sp = new SoundPool( 10 , StreamType.MUSIC, 5 ); //载入 soundPoolMap = new HashMap<integer, integer= "" >(); soundPoolMap.put( 0 , soundPool.load(context, R.raw.sound1, 1 )); //播放 soundPool.play(soundPoolMap.get( 0 ), 1 , 1 , 0 , 0 , 1 );</integer,> |
各个函数中参数的具体意义不是重点,这里就不做介绍了,各位自己去查一下。我想要说的是,如果各位在主线程中直接这么做,很可能会报错(Sample X Not Ready),这是为什么呢?
原来,SoundPool.load()这个函数是立即返回的,也就是说,不管载入好了没有,这个函数都会返回,但是实际上非常有可能声音尚未载入结束,那应该怎么做呢?实际上,系统在load完成之后,会发送广播,这时候才能播放。又因为这是一个费时的操作,所以最好新建一个线程来实现。下面是我的实现方案:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
|
Handler handler; static SoundPool soundPool; static HashMap<integer, integer= "" > soundPoolMap; String TAG = "wtianok" ; @Override protected void onCreate(Bundle savedInstanceState) { super .onCreate(savedInstanceState); setContentView(R.layout.activity_main); if (savedInstanceState == null ) { getSupportFragmentManager().beginTransaction() .add(R.id.container, new PlaceholderFragment()).commit(); } soundPool = new SoundPool( 10 , AudioManager.STREAM_MUSIC, 5 ); soundPoolMap = new HashMap<integer, integer= "" >(); // 载入soundPool new LoadSoundPoolThread().run( this ); handler = new Handler() { public void handleMessage(android.os.Message msg) { if (msg.what == MessageType.SOUND_POOL_READY.ordinal()) { Log.d(TAG, "finish load message received" ); //do somethind } } } } private class LoadSoundPoolThread extends Thread { public void run(Context context) { Log.d(TAG, "start to load soundpool" ); soundPool.setOnLoadCompleteListener( new MyOnLoadCompleteListener()); soundPool.load(context, R.raw.sound1, 1 ); soundPool.load(context, R.raw.sound2, 1 ); soundPool.load(context, R.raw.sound3, 1 ); soundPool.load(context, R.raw.sound4, 1 ); soundPool.load(context, R.raw.sound5, 1 ); soundPool.load(context, R.raw.sound6, 1 ); soundPool.load(context, R.raw.sound7, 1 ); } private class MyOnLoadCompleteListener implements OnLoadCompleteListener { int count = 1 ; @Override public void onLoadComplete(SoundPool soundPool, int sampleId, int status) { //注意这里的形参 Log.d(TAG, "load " + count + " sound" ); soundPoolMap.put(count, sampleId); if (count == 7 ) { Log.d(TAG, "finish load soundpool" ); MainActivity.soundPool = soundPool; sendMsg(MessageType.SOUND_POOL_READY); count = 1 ; } count++; } } } private void sendMsg(MessageType micTestStartRecord) { Message message = Message.obtain(); message.what = micTestStartRecord.ordinal(); handler.sendMessage(message); }</integer,></integer,> |
需要注意的是,在完成载入之后,我使用了下面这一句:
1
|
MainActivity.soundPool = soundPool; |
如果不加这一句,我在主线程中play的时候,依然会报Sample X Not Ready错误,而如果在收到广播之后调用play是没有问题的。仔细看,发现在接收广播的onLoadComplete()函数中,实际上调用的是函数的局部参数soundPool,而不是全局的soundPool,但是从代码来看,load的时候调用的确实全局的soundPool,那为什么用全局的soundPool就不能播放呢?我到现在还是没有想通,但是项目还是要继续,只好加了上面那一句,将全局的soundPool重新赋值,事实证明,这样做可以。
结伴旅游,一个免费的交友网站:www.jieberu.com
推推族,免费得门票,游景区:www.tuituizu.com
Android中播放声音的更多相关文章
- Android中Broadcast Receiver组件具体解释
BroadcastReceiver(广播接收器)是Android中的四大组件之中的一个. 以下是Android Doc中关于BroadcastReceiver的概述: ①广播接收器是一个专注于接收广播 ...
- 【Android中Broadcast Receiver组件具体解释
】
BroadcastReceiver(广播接收器)是Android中的四大组件之中的一个. 以下是Android Doc中关于BroadcastReceiver的概述: ①广播接收器是一个专注于接收广播 ...
- Android中四大组件总结
android四大组件分别为activity.service.content provider.broadcast receiver. 一.android四大组件详解 1.activity (1)一个 ...
- Android中的LinearLayout布局
LinearLayout : 线性布局 在一般情况下,当有很多控件需要在一个界面列出来时,我们就可以使用线性布局(LinearLayout)了, 线性布局是按照垂直方向(vertical)或水平方向 ...
- Android中BroadcastReceiver的两种注册方式(静态和动态)详解
今天我们一起来探讨下安卓中BroadcastReceiver组件以及详细分析下它的两种注册方式. BroadcastReceiver也就是"广播接收者"的意思,顾名思义,它就是用来 ...
- Android中使用ExpandableListView实现微信通讯录界面(完善仿微信APP)
之前的博文<Android中使用ExpandableListView实现好友分组>我简单介绍了使用ExpandableListView实现简单的好友分组功能,今天我们针对之前的所做的仿微信 ...
- Android中ListView实现图文并列并且自定义分割线(完善仿微信APP)
昨天的(今天凌晨)的博文<Android中Fragment和ViewPager那点事儿>中,我们通过使用Fragment和ViewPager模仿实现了微信的布局框架.今天我们来通过使用Li ...
- Android中Fragment和ViewPager那点事儿(仿微信APP)
在之前的博文<Android中使用ViewPager实现屏幕页面切换和引导页效果实现>和<Android中Fragment的两种创建方式>以及<Android中Fragm ...
- Android中Fragment与Activity之间的交互(两种实现方式)
(未给Fragment的布局设置BackGound) 之前关于Android中Fragment的概念以及创建方式,我专门写了一篇博文<Android中Fragment的两种创建方式>,就如 ...
随机推荐
- Hive-Container killed by YARN for exceeding memory limits. 9.2 GB of 9 GB physical memory used. Consider boosting spark.yarn.executor.memoryOverhead.
Caused by: org.apache.spark.SparkException: Job aborted due to stage failure: Task times, most recen ...
- Linux 下面根据端口号 查询 可执行程序的路劲的方法
1. 安装上lsof 的包 2. 使用 lsof 命令查看相关进程 lsof -i: 效果为: 3. 根据/proc 的目录查看可执行目录的文件位置 ll /proc/procid # procid ...
- centos 防火墙 iptables firewalld SELinux
参考 Centos7 只启用iptables 禁用firewalld功能 java.net.NoRouteToHostException: 没有到主机的路由 相关内容 centos7 中才开始引用fi ...
- JavaSE--类与对象
一.类 类是具有相同特性(数据元素)和行为(功能)的对象的抽象就是类.因此,对象的抽象是类,类的具体化就是对象,也可以说类的实例是对象,类实际上就是一种数据类型.类具有属性,它是对象的状态的抽象,用数 ...
- Python 进阶篇
作者:武沛齐 出处:http://www.cnblogs.com/wupeiqi/articles/5246483.html Model 到目前为止,当我们的程序涉及到数据库相关操作时,我们一般都会这 ...
- luogu P4382 [九省联考2018]劈配
luogu 我记得我第一次做这道题的时候屁都不会qwq 先考虑第一问,暴力是依次枚举每个人,然后从高到低枚举志愿,枚举导师,能选就选.但是可以发现前面的人选的导师可能会导致后面的人本来可以选到这个志愿 ...
- Css布局 响应式布局介绍
1. 概念: 写一套css样式可以同时适配多个终端,是为解决移动互联网诞生的. 2. 作用: 面对不同的分辨率设备灵活性强,能够快捷解决多设备显示适应问题 3. 原理 媒体查询 ① 外联式媒体查询语法 ...
- BTE增强解析
原理:转载http://blog.csdn.net/wbin9752/article/details/7954663 BTEs(Business Transaction Events),是SAP的一种 ...
- 日语能力测试N1、N2级听力必备核心词汇—头发篇
日语能力测试N1.N2级听力必备核心词汇—头发篇 要想在短时间内迅速提高日语听力能力的水平,除了每天练习(用2倍的速度)真题之外,掌握听力的核心词汇也是一个必要的好方法. 髪(かみ)--头发髪型(かみ ...
- Redis5以上版本伪集群搭建(高可用集群模式)
redis集群需要至少要三个master节点,我们这里搭建三个master节点,并且给每个master再搭建一个slave节点,总共6个redis节点,这里用一台机器(可以多台机器部署,修改一下ip地 ...