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中播放声音的更多相关文章

  1. Android中Broadcast Receiver组件具体解释

    BroadcastReceiver(广播接收器)是Android中的四大组件之中的一个. 以下是Android Doc中关于BroadcastReceiver的概述: ①广播接收器是一个专注于接收广播 ...

  2. 【Android中Broadcast Receiver组件具体解释 】

    BroadcastReceiver(广播接收器)是Android中的四大组件之中的一个. 以下是Android Doc中关于BroadcastReceiver的概述: ①广播接收器是一个专注于接收广播 ...

  3. Android中四大组件总结

    android四大组件分别为activity.service.content provider.broadcast receiver. 一.android四大组件详解 1.activity (1)一个 ...

  4. Android中的LinearLayout布局

    LinearLayout : 线性布局 在一般情况下,当有很多控件需要在一个界面列出来时,我们就可以使用线性布局(LinearLayout)了,  线性布局是按照垂直方向(vertical)或水平方向 ...

  5. Android中BroadcastReceiver的两种注册方式(静态和动态)详解

    今天我们一起来探讨下安卓中BroadcastReceiver组件以及详细分析下它的两种注册方式. BroadcastReceiver也就是"广播接收者"的意思,顾名思义,它就是用来 ...

  6. Android中使用ExpandableListView实现微信通讯录界面(完善仿微信APP)

    之前的博文<Android中使用ExpandableListView实现好友分组>我简单介绍了使用ExpandableListView实现简单的好友分组功能,今天我们针对之前的所做的仿微信 ...

  7. Android中ListView实现图文并列并且自定义分割线(完善仿微信APP)

    昨天的(今天凌晨)的博文<Android中Fragment和ViewPager那点事儿>中,我们通过使用Fragment和ViewPager模仿实现了微信的布局框架.今天我们来通过使用Li ...

  8. Android中Fragment和ViewPager那点事儿(仿微信APP)

    在之前的博文<Android中使用ViewPager实现屏幕页面切换和引导页效果实现>和<Android中Fragment的两种创建方式>以及<Android中Fragm ...

  9. Android中Fragment与Activity之间的交互(两种实现方式)

    (未给Fragment的布局设置BackGound) 之前关于Android中Fragment的概念以及创建方式,我专门写了一篇博文<Android中Fragment的两种创建方式>,就如 ...

随机推荐

  1. [转帖]VMWare官网:无法关闭 ESXi 主机上的虚拟机 (1014165)

    无法关闭 ESXi 主机上的虚拟机 (1014165) https://kb.vmware.com/s/article/1014165?lang=zh_CN Last Updated: 4/17/20 ...

  2. 解决react-native 运行报错:Entry, ":CFBundleIdentifier", Does Not Exist

    首次运行react-native命令很可能报这样的错误,网上也有一堆人写出了解决方案,但可能每个人出错的原因都不一样,建议把ios目录在xcode中运行下,可以看到报错原因. 我的报错原因是因为808 ...

  3. [codeforces1234F]Yet Another Substring Reverse

    题目链接 大致题意为将某个子串进行翻转后,使得不包含相同字符的字符子串长度最长.只能翻转一次或零次. 设一个子串的状态为包含字符的二进制.如子串为$abacd$,则状态为$00000000000000 ...

  4. Python 入门之 内置模块 -- re模块

    Python 入门之 内置模块 -- re模块 1.re 模块 (1)什么是正则? 正则就是用一些具有特殊含义的符号组合到一起(称为正则表达式)来描述字符或者字符串的方法.或者说:正则就是用来描述一类 ...

  5. Python字符串和正则表达式中的反斜杠('\')问题

    在Python普通字符串中 在Python中,我们用'\'来转义某些普通字符,使其成为特殊字符,比如 In [1]: print('abc\ndef') # '\n'具有换行的作用 abc defg ...

  6. 细说vue axios登录请求拦截器

    当我们在做接口请求时,比如判断登录超时时候,通常是接口返回一个特定的错误码,那如果我们每个接口都去判断一个耗时耗力,这个时候我们可以用拦截器去进行统一的http请求拦截. 1.安装配置axios cn ...

  7. Gym 101466(完整)

    题目链接 :点击此处 ## Problem A 题意: 给你n个数,重定义两个数之间的加法不进位,求这些数中两个数相加的最大值和最小值. 题解: 字典树.我们首先将前i-1为放入字典树中,然后在查询第 ...

  8. 将本地的java项目提交到github出错解决

    1.我们新建一个了java项目后,需要提交到github进行版本控制 2.如果此时github中的仓库不为空,我们在本地使用git push提交时会报以下错误, ! [rejected]       ...

  9. 硬盘安装ubuntu遇到的问题

    终于把这个系统给装上了,陆陆续续弄了4,5天(崩溃...),一直一来都是用U盘来装ubuntu的,挺简单的,但是这个主机识别不了U盘不知道为什么...这个问题又是百度又是Google最终找不到原因只好 ...

  10. Ubuntu18.04 安装netstat

    关键命令 sudo apt-get install net-tools 参考博客:https://www.cnblogs.com/2sheep2simple/p/10611650.html