【Android】MediaPlayer之音频播放

作者:邹峰立,微博:zrunker,邮箱:[email protected],微信公众号:书客创作,个人平台:www.ibooker.cc。

本文选自书客创作平台第45篇文章。阅读原文 。

【Android】MediaPlayer之音频播放_第1张图片
书客创作

如果对MediaPlayer不理解推荐这篇文章【Android】MediaPlayer生命周期分析。

要想利用MediaPlayer实现音频的播放,首先要对MediaPlayer进行初始化工作,得到MediaPlayer对象,在通过MediaPlayer进行相应的操作。

一般过程:初始化MediaPlayer - 加载媒体源 - 准备 - 开始播放

MediaPlayer mediaPlayer = new MediaPlayer();
mediaPlayer.setDataSource("...");
mediaPlayer.prepare();
mediaPlayer.start();

MediaPlayer支持多种不同的媒体源: 本地资源、内部的URI,比如一个你可能会从ContentResolver获取的uri、外部URL(流)。

1、raw文件中媒体源:假如res/raw文件中包含一个sound_music.mp3文件。

MediaPlayer mediaPlayer = MediaPlayer.create(this, R.raw.sound_music);

2、assets文件中媒体源:假如在assets中包含一个sound_music.mp3文件。

try {
    AssetFileDescriptor fd = getAssets().openFd("sound_music.mp3");
    MediaPlayer mediaPlayer = new MediaPlayer();
    mediaPlayer.setDataSource(fd.getFileDescriptor(), fd.getStartOffset(), fd.getLength());
} catch (IOException e) {
    e.printStackTrace();
}

3、SD卡中媒体源:假如在SD卡中包含一个sound_music.mp3文件。

try {
    MediaPlayer mediaPlayer = new MediaPlayer();
    String path = "/sdcard/sound_music.mp3";
    mediaPlayer.setDataSource(path);
} catch (IOException e) {
    e.printStackTrace();
}

4、网络资源:假如有一个网络资源http://ibooker.cc/ibooker/musics/sound_music.mp3。

MediaPlayer mediaPlayer = new MediaPlayer();
// 方式一
//    Uri uri = Uri.parse("http://ibooker.cc/ibooker/musics/sound_music.mp3");
//    mediaPlayer.setDataSource(this, uri);

// 方式二
mediaPlayer.setDataSource("http://ibooker.cc/ibooker/musics/sound_music.mp3");

MediaPlayer常用方法

int getCurrentPosition();// 得到当前播放位置(ms)
int getDuration();// 得到文件的时间(ms)
void setLooping(boolean var1);// 设置是否循环播放
boolean isLooping();// 是否循环播放
boolean isPlaying();// 是否正在播放
void pause();// 暂停
void prepare();// 同步准备
void prepareAsync();// 异步准备
void release();// 释放MediaPlayer对象
void reset();// 重置MediaPlayer对象
void seekTo(int msec);// 指定播放位置(以毫秒为单位)
void setDataSource(String path);// 设置播放资源
void setScreenOnWhilePlaying(boolean screenOn);// 设置播放的时候一直让屏幕变亮
void setWakeMode(Context context, int mode);// 设置唤醒模式
void setVolume(float leftVolume, float rightVolume);// 设置音量,参数分别表示左右声道声音大小,取值范围为0~1
void start();// 开始播放
void stop();// 停止播放

MediaPlayer常用事件监听

播放出错监听

MediaPlayer.setOnErrorListener(new MediaPlayer.OnErrorListener() {
    @Override
    public boolean onError(MediaPlayer mediaPlayer, int i, int i1) {
        return false;
    }
});

播放完成监听

MediaPlayer.setOnCompletionListener(new MediaPlayer.OnCompletionListener() {
    @Override
    public void onCompletion(MediaPlayer mediaPlayer) {
        // todo
    }
});

网络流媒体缓冲监听

MediaPlayer.setOnBufferingUpdateListener(new MediaPlayer.OnBufferingUpdateListener() {
    @Override
    public void onBufferingUpdate(MediaPlayer mediaPlayer, int i) {
        // i 0~100
        Log.d("Progress:", "缓存进度" + i + "%");
    }
});

准备Prepared完成监听

MediaPlayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
    @Override
    public void onPrepared(MediaPlayer mediaPlayer) {
        // todo
    }
});

进度调整完成SeekComplete监听,主要是配合seekTo(int)方法

MediaPlayer.setOnSeekCompleteListener(new MediaPlayer.OnSeekCompleteListener() {
    @Override
    public void onSeekComplete(MediaPlayer mediaPlayer) {
        // todo
    }
});

使用锁

当音频在后台播放时,设备可能进入到休眠状态,此时系统会关闭一些不必要的资源,包括CPU和wifi等等...如果想要在后台播放或者缓冲音乐并且不想被系统干扰必须使用锁,并且在paused或者stopped状态下释放它。

// 设置设备进入锁状态模式-可在后台播放或者缓冲音乐-CPU一直工作
mediaPlayer.setWakeMode(this, PowerManager.PARTIAL_WAKE_LOCK);

同时需要在清单文件AndroidManifest.xml中添加权限


上面代码只会确保cpu一直工作,如果你使用wifi播放流媒体,你还需要持有wifi锁

// 如果你使用wifi播放流媒体,你还需要持有wifi锁
WifiManager.WifiLock wifiLock = ((WifiManager) getApplicationContext().getSystemService(Context.WIFI_SERVICE)).createWifiLock(WifiManager.WIFI_MODE_FULL, "wifilock");
wifiLock.acquire();

当你停止播放或者不再需要连接网络释放

wifiLock.release();

处理音频焦点

android是一个支持多任务的系统,这给使用audio的程序带来了一定的风险,因为可能几个程序会来竞争音频输出设备,在android2.2之前没有内置的机制来处理这个问题,导致体验很差。比如用户在听音乐时,另一个应用需要向用户提示一个非常重要的通知,但音乐声盖过了通知音,导致用户错失了最佳接收提示的时间,为了协调设备的音频输出,android提出了Audio Focus机机制,获取audio focus必须调用AudioManager的requestAudioFocus()方法。

// 处理音频焦点-处理多个程序会来竞争音频输出设备
AudioManager audioManager = (AudioManager) getSystemService(Context.AUDIO_SERVICE);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
    // 征对于Android 8.0+
    AudioFocusRequest audioFocusRequest = new AudioFocusRequest.Builder(AudioManager.AUDIOFOCUS_GAIN_TRANSIENT)
                    .setOnAudioFocusChangeListener(focusChangeListener).build();
    audioFocusRequest.acceptsDelayedFocusGain();
    audioManager.requestAudioFocus(audioFocusRequest);
} else {
    // 小于Android 8.0
    int result = audioManager.requestAudioFocus(focusChangeListener, AudioManager.STREAM_MUSIC, AudioManager.AUDIOFOCUS_GAIN);
    if (result != AudioManager.AUDIOFOCUS_REQUEST_GRANTED) {
        // could not get audio focus.
    }
}

focusChange参数值如下

    1. AUDIOFOCUS_GAIN:获取audio focus
    1. AUDIOFOCUS_LOSS:失去audio focus很长一段时间,必须停止所有的audio播放,清理资源
    1. AUDIOFOCUS_ LOSS_TRANSIENT:暂时失去audio focus,但是很快就会重新获得,在此状态应该暂停所有音频播放,但是不能清除资源
    1. AUDIOFOCUS_ LOSS_TRANSIENT _CAN_DUCK:暂时失去 audio focus,但是允许持续播放音频(以很小的声音),不需要完全停止播放。
AudioManager.OnAudioFocusChangeListener focusChangeListener = new AudioManager.OnAudioFocusChangeListener() {
    @Override
    public void onAudioFocusChange(int focusChange) {
        switch (focusChange) {
            case AudioManager.AUDIOFOCUS_GAIN:
                // 获取audio focus
                if (mediaPlayer == null)
                    mediaPlayer = new MediaPlayer();
                else if (!mediaPlayer.isPlaying())
                    mediaPlayer.start();
                mediaPlayer.setVolume(1.0f, 1.0f);
                break;
            case AudioManager.AUDIOFOCUS_LOSS:
                // 失去audio focus很长一段时间,必须停止所有的audio播放,清理资源
                if (mediaPlayer.isPlaying())
                    mediaPlayer.stop();
                mediaPlayer.release();
                mediaPlayer = null;
                break;
            case AudioManager.AUDIOFOCUS_LOSS_TRANSIENT:
                // 暂时失去audio focus,但是很快就会重新获得,在此状态应该暂停所有音频播放,但是不能清除资源
                if (mediaPlayer.isPlaying())
                    mediaPlayer.pause();
                break;
            case AudioManager.AUDIOFOCUS_LOSS_TRANSIENT_CAN_DUCK:
                // 暂时失去 audio focus,但是允许持续播放音频(以很小的声音),不需要完全停止播放。
                if (mediaPlayer.isPlaying())
                    mediaPlayer.setVolume(0.1f, 0.1f);
                break;
        }
    }
};

处理AUDIO_ BECOMING_NOISY意图

当用户插着耳机听音乐突然拔掉耳机,如果没有对以上意图进行处理,音频将会通过外部扬声器来播放音频,这也许不会是用户想要的,所以我们必须对此处理。

可以使用动态广播broadcastReceiver

broadcastReceiver = new BroadcastReceiver() {
    @Override
    public void onReceive(Context context, Intent intent) {
        if (AudioManager.ACTION_AUDIO_BECOMING_NOISY.equals(intent.getAction())) {
            // 拔掉耳机时候进行相应的操作
        }
    }
};
registerReceiver(broadcastReceiver, new IntentFilter(AudioManager.ACTION_AUDIO_BECOMING_NOISY));

然后在页面onDestory方法中注销广播即可。

@Override
protected void onDestroy() {
    super.onDestroy();
    unregisterReceiver(broadcastReceiver);
}

当然也可以使用静态广播

1、在manifest文件中注册一个广播


        
                            
      

2、注册MusicIntentReceiver 广播来处理这个intent

public class MusicIntentReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context ctx, Intent intent) {
        if (intent.getAction().equals(android.media.AudioManager.ACTION_AUDIO_BECOMING_NOISY)) {
            // signal your service to stop playback
            // (via an Intent, for instance)
        }
    }
}

检索媒体文件

// 通过ContentResolver来获取外部媒体文件
ContentResolver contentResolver = getContentResolver();
Uri uri = android.provider.MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
Cursor cursor = contentResolver.query(uri, null, null, null, null);
if (cursor == null) {
    // query failed, handle error.
} else if (!cursor.moveToFirst()) {
    // no media on the device
} else {
    int titleColumn = cursor.getColumnIndex(android.provider.MediaStore.Audio.Media.TITLE);
    int idColumn = cursor.getColumnIndex(android.provider.MediaStore.Audio.Media._ID);
    do {
        long thisId = cursor.getLong(idColumn);
        String thisTitle = cursor.getString(titleColumn);
        // ...process entry...
    } while (cursor.moveToNext());
}

// 配合MediaPlayer使用
try {
    long id = /* retrieve it from somewhere */;
    Uri contentUri = ContentUris.withAppendedId(android.provider.MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, thisId);
    mediaPlayer = new MediaPlayer();
    mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
    mediaPlayer.setDataSource(getApplicationContext(), contentUri);
    // ...prepare and start...
} catch (IOException e) {
    e.printStackTrace();
}

简易播放音乐器案例

常见的音乐播放器都有,上一首、下一首、播放、暂停、停止等功能,那接下来就用一个简易的音乐播放器来说明MediaPlayer的使用。首先看一下布局效果图:

【Android】MediaPlayer之音频播放_第2张图片
布局效果图



    

        

        

        

        

        

    

    


在该实例中以加载网络资源为例,这里定义一个字符串数据保存网络URL。

private String[] musics = {"http://ibooker.cc/ibooker/musics/1234.mp3",
            "http://ibooker.cc/ibooker/musics/2345.mp3"}; // 设置音频资源(网络)

在程序一来是,需要对MediaPlayer进行初始化工作,所以定义一个全局变量MediaPlayer。

// 初始化MediaPlayer
private void initMediaPlayer() {
    if (mediaPlayer == null)
        mediaPlayer = new MediaPlayer();
    // 设置音量,参数分别表示左右声道声音大小,取值范围为0~1
    mediaPlayer.setVolume(0.5f, 0.5f);
    // 设置是否循环播放
    mediaPlayer.setLooping(false);
}

当点击播放按钮,程序要马上播放相应的音乐文件,因为是加载网络资源,所以在这里采用prepareAsync进行准备,同时设置OnPreparedListener监听是否准备完成。

// 播放
private void play() {
    try {
        if (mediaPlayer == null)
            initMediaPlayer();
        if (isPause) {
            mediaPlayer.start();
            updateDescTv();
        } else {
            // 重置mediaPlayer
            mediaPlayer.reset();
            // 重新加载音频资源
//         Uri uri = Uri.parse(musics[current_item]);
//         mediaPlayer.setDataSource(this, uri);
            mediaPlayer.setDataSource(musics[current_item]);
            // 准备播放(同步)-预期准备,因为setDataSource()方法之后,MediaPlayer并未真正的去装载那些音频文件,需要调用prepare()这个方法去准备音频
//         mediaPlayer.prepare();
            // 准备播放(异步)
            mediaPlayer.prepareAsync();
        }
    } catch (IOException e) {
        e.printStackTrace();
    }
}

注:isPause是用来标记当前播放是否处于暂停状态。

// 异步准备Prepared完成监听
mediaPlayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
    @Override
    public void onPrepared(MediaPlayer mediaPlayer) {
        // 开始播放
        mediaPlayer.start();
        updateDescTv();
    }
});

updateDescTv();方法是用来显示播放进度。这里是通过开启一个线程来时时监听播放进度。

// 开启线程,修改descTv
private void updateDescTv() {
    Thread thread = new Thread(new Runnable() {
        @Override
        public void run() {
            try {
                while (mediaPlayer != null && mediaPlayer.isPlaying()) {// 判断音频是否正在播放
                    myHandler.sendEmptyMessage(100);
                    Thread.sleep(1000);
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    });
    if (executorService == null || executorService.isShutdown())
        executorService = Executors.newSingleThreadExecutor();
    executorService.execute(thread);
}

MyHandler myHandler = new MyHandler(this);

private static class MyHandler extends Handler {
    private WeakReference mActivity;

    MyHandler(Activity activity) {
        mActivity = new WeakReference<>(activity);
    }

    @Override
    public void handleMessage(Message msg) {
        super.handleMessage(msg);
        MainActivity currentActivity = (MainActivity) mActivity.get();

        switch (msg.what) {
            case 100:
                if (currentActivity.mediaPlayer != null)
                    currentActivity.descTv.setText("播放进度:" + currentActivity.mediaPlayer.getCurrentPosition() * 100 / currentActivity.mediaPlayer.getDuration() + "%");
                break;
        }
    }
}

要想实现下一首和上一首的功能,还需要定义一个变量current_item标记当前播放哪一首。

// 下一首
private void nextMusic() {
    current_item++;
    if (current_item >= musics.length)
        current_item = 0;
    play();
}

// 上一首
private void preMusic() {
    current_item--;
    if (current_item < 0)
        current_item = musics.length - 1;
    play();
}

当点击暂停时

if (mediaPlayer.isPlaying()) {
    isPause = true;
    mediaPlayer.pause();
}

当点击停止时

if (mediaPlayer.isPlaying())
    mediaPlayer.reset();

最后在页面onStop()中回收资源即可

@Override
protected void onStop() {
    super.onStop();
    // 释放mediaPlayer
    if (mediaPlayer != null) {
        mediaPlayer.stop();
        mediaPlayer.release();
        mediaPlayer = null;
    }
}

Github地址
阅读原文


【Android】MediaPlayer之音频播放_第3张图片
微信公众号:书客创作

你可能感兴趣的:(【Android】MediaPlayer之音频播放)