Android多媒体相关的API,网上基本都能找到很多相关的文章,使用起来也很简单,一直在犹豫要不要写这方面的内容,后来决定还是写一写,一方面算是一个归纳总结,另一方面,也方便以后查阅。这一篇就写一下MediaPlayer。

状态图详解

下图是一个MediaPlayer的生命周期和状态。其中,椭圆代表MediaPlayer可能驻留的状态,弧线表示MediaPlayer的播放控制操作。这里有两种类型的弧线,单箭头弧线代表同步方法调用,双箭头弧线代表异步方法调用。

1、新创建的MediaPlayer对象、或者调用了reset()方法的MediaPlayer对象,都处于Idle状态,这两种方法得到的对象,有一个微小但十分重要的差别。

处于Idle状态时,调用 getCurrentPosition(), getDuration(), getVideoHeight(), getVideoWidth(), setAudioStreamType(), setLooping(), setVolume(), pause(), start(), stop(), seekTo(), prepare(), prepareAsync()方法都会报错。新创建的MediaPlayer对象,调用以上方法,无法接收到注册的OnErrorListener.onError()回调;调用reset()方法的MediaPlayer对象可以接收到回调。

MediaPlayer不再被使用时,应立即调用release()方法来释放资源,资源可能包括硬件加速组件的单态固件,若没有调用release()方法可能会导致之后的MediaPlayer对象实例无法使用这种单态硬件资源,导致异常。

一旦MediaPlayer对象进入了End状态,将不能再被使用,也没有办法再迁移到其他状态。

2、由于种种原因,一些操作可能会失败,如不支持的格式/分辨率太高/流超时等,还有编程错误(比如在无效状态下调用某个操作),此时会回调OnErrorListener.onError()方法(需客户端提前注册listener)。一旦发生错误,MediaPlayer对象会进入Error状态,此时可以调用reset()方法把这个对象恢复到Idle状态。

在不合法的状态下调用一些方法,如prepare()、prepareAsync()和setDataSource()等会抛出ILlegalStateException异常。

3、Idle状态下,调用 setDataSource()方法会迁移到Initialized状态,非Idle状态下调用此方法会报 ILlegalStateException异常。注意,setDataSource()方法可能会抛出IOException异常。

4、调用prepare()、prepareAsync()方法可以迁移到Prepared状态,该状态下才可以进行基本播放操作。

异步的prepareAsync()方法需要通过OnPrepareListener.onPrepared()监听准备是否完成,Preparing是一个中间状态,如果在此状态下调用任何影响播放功能的方法,最终的运行结果都是未知的。

在不合适的状态下调用prepare()和prepareAsync()方法会抛出ILlegalStateException异常。

5、调用start()方法成功返回后,会迁移到Started状态,isPlaying()方法返回是否处于Started状态。迁移到Started状态时,可以通过OnBufferingUpdateListener.onBufferingUpdate()回调得知。

Started状态下调用start()方法没有影响。

6、调用pause()方法并返回时,会迁移到Paused状态。注意,Started与Paused状态的转换在内部的播放引擎中是异步的,所以isPlaying()可能会延时更新,如果是播放网络流媒体,这个延时可能会有几秒。

Paused状态下调用pause()方法没有影响。

7、除了Idle、Initialized状态,其它状态下都可以调用stop()迁移到Stopped状态,Stopped状态下调用stop()方法没有影响。

8、seekTo()方法可以调整播放位置,seekTo()方法是异步的,尤其是播放网络流媒体时延时很明显。实际定位完成后,通过OnSeekComplete.onSeekComplete()通知。

“活动状态”(Prepared、Started、Paused、PlaybackCompleted状态)下都可以调用seekTo()方法。

9、迁移到PlaybackCompleted状态后,如果通过setLooping()方法开启了循环模式,会重新进入到Started状态,并且不会回调OnCompletion.onCompletion()方法、如果没有开启循环,就会回调这个方法。

PlaybackCompleted状态下调用start()方法会迁移到Started状态。

各方法的调用状态

除了下面几个方法调用时需要特别注意状态的判断,其余常用方法,基本所有状态都是OK的,或者即便状态不对也不会报错。如果对某个方法调用有疑问,查阅API文档,下面只列出一些常用的、需要注意状态的方法。

下面这几个方法需要注意下:

1、setDataSource()
有效状态:Idle
调用结果:调用成功,会迁移到Initialized状态
无效状态:报IllegalStateException异常

2、prepare()、prepareAsync()
有效状态:Initialized/Stopped
调用结果:调用成功,会迁移到Prepared/Preparing状态
无效状态:报IllegalStateException异常

3、pause()
有效状态:Started/Paused
调用结果:调用成功,会迁移到Paused状态
无效状态:player进入Error状态

4、start()
有效状态:Prepared/Started/Paused/PlaybackCompleted
调用结果:调用成功,会迁移到Started状态
无效状态:player进入Error状态

5、stop()
有效状态:Prepared/Started/Stopped/Paused/PlaybackCompleted
调用结果:调用成功,会迁移到Stopped状态
无效状态:player进入Error状态

6、seekTo()
有效状态:Prepared/Started/Paused/PlaybackCompleted
调用结果:调用成功,不会改变player的状态
无效状态:player进入Error状态

使用demo

使用之前先整理一下大概要用到播放操作的方法,每个方法的有效状态是怎样。

大概要用到setDataSource()、prepare()、start()、pause()、seekTo()这5个方法,stop()方法一般不用,不播放的时候最好调release()释放资源。其中setDataSource()和prepare()基本就是初始化的时候连续调用,不太需要注意状态,所以剩下需要注意的也就start()、pause()和seekTo()这3个方法。

排除stop()方法和Stopped状态后,从“不可操作”到“可操作”的分界点就在Prepared状态,所以我们可以使用一个变量hasPrepared来标记是否可操作,start()和seekTo()方法在“可操作”状态下都是可以正常调用的。

剩下的就是pause()方法,只能在Started和Paused状态下调用,可以使用一个变量canPause来标记是否可以调用pause()方法。实际上,Prepared和PlaybackCompleted状态在程序中也基本是个瞬时状态,基本不会停留。一旦Prepared,程序就会调用start()方法进行播放;一旦PlaybackCompleted,就会进行下一个曲目的初始化、准备、播放。

那么下面就看一下使用方法吧:

public class MyPlayer implements MediaPlayer.OnPreparedListener, MediaPlayer.OnErrorListener, MediaPlayer.OnCompletionListener {private MediaPlayer mPlayer;private boolean hasPrepared;private void initIfNecessary() {if (null == mPlayer) {mPlayer = new MediaPlayer();mPlayer.setOnErrorListener(this);mPlayer.setOnCompletionListener(this);mPlayer.setOnPreparedListener(this);}}public void play(Context context, Uri dataSource) {hasPrepared = false; // 开始播放前讲Flag置为不可操作initIfNecessary(); // 如果是第一次播放/player已经释放了,就会重新创建、初始化try {mPlayer.reset();mPlayer.setDataSource(context, dataSource); // 设置曲目资源mPlayer.prepareAsync(); // 异步的准备方法} catch (IOException e) {e.printStackTrace();}}public void start() {// release()会释放player、将player置空,所以这里需要判断一下if (null != mPlayer && hasPrepared) {mPlayer.start();}}public void pause() {if (null != mPlayer && hasPrepared) {mPlayer.pause();}}public void seekTo(int position) {if (null != mPlayer && hasPrepared) {mPlayer.seekTo(position);}}// 对于播放视频来说,通过设置SurfaceHolder来设置显示Surface。这个方法不需要判断状态、也不会改变player状态public void setDisplay(SurfaceHolder holder) {if (null != mPlayer) {mPlayer.setDisplay(holder);}}public void release() {hasPrepared = false;mPlayer.stop();mPlayer.release();mPlayer = null;}@Overridepublic void onPrepared(MediaPlayer mp) {hasPrepared = true; // 准备完成后回调到这里start();}@Overridepublic void onCompletion(MediaPlayer mp) {hasPrepared = false;// 通知调用处,调用play()方法进行下一个曲目的播放}@Overridepublic boolean onError(MediaPlayer mp, int what, int extra) {hasPrepared = false;return false;}
}

MediaPlayer详解和使用相关推荐

  1. MediaPlayer详解

    Android开发之MdiaPlayer详解 MediaPlayer类可用于控制音频/视频文件或流的播放,我曾在<Android开发之基于Service的音乐播放器>一文中介绍过它的使用. ...

  2. Android的媒体播放器------简易音乐播放器(详解)

    一.案例效果 二.欢迎界面的设计与功能 2.1 .案例效果 设计一个倒计时自动跳转的页面 2.2. 布局界面 activity_welcome.xml 参考代码: <?xml version=& ...

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

    为什么80%的码农都做不了架构师?>>>     李华明Himi 原创,转载务必在明显处注明: 转载自 [黑米GameDev街区] 原文链接:  http://www.himigam ...

  4. android gridview控件使用详解_作为Android 开发者该如何进阶?

    经常在简书和微信上收到一些同学的私信,说自己马上毕业或者已经毕业一年,从事Android开发相关的工作,现在不知道要学习什么东西了.或者说自己也在摸索着学习,但是不知道学习的路线对不对,感觉很迷茫,想 ...

  5. Android Lifecycle 生命周期组件详解

    转载请标明出处:https://blog.csdn.net/zhaoyanjun6/article/details/99695779 本文出自[赵彦军的博客] 一.Lifecycle简介 为什么要引进 ...

  6. Android JNI作用及其详解

    Android JNI作用及其详解 Java Native Interface (JNI)标准是Java平台的一部分,它允许Java代码和其他语言写的代码进行交互.JNI 是本地编程接口,它使得在 J ...

  7. Android BINDER详解

    1.   进程间通信的本质(2个进程) 用户空间的进程如果想相互通信, 必须经过内核, 因为不同进程的用户地址空间是独立的, 但是共享同一个内核空间. 内核为了支持进程间通信, 一般会有一个驱动, 以 ...

  8. 桌面widget详解(四)——桌面音乐播放器(实战)

    前言:这将是这个系列的最后一篇了,我写这几篇文章也是累的快不行了,再写就真的要吐了,言归正转,前面三篇已经把widget中涉及到的基本知识基本上讲完了,今天我们就做一个小例子,看看桌面音乐播放器wid ...

  9. android中oncreate方法,android开发之onCreate( )方法详解

    这里我们只关注一句话:This is where you should do all of your normal static set up.其中我们只关注normal static, normal ...

最新文章

  1. python异常值删除_python删除有异常值
  2. leetCode刷题 2. 两数相加
  3. python--numpy pad函数使用
  4. C的指针疑惑:C和指针8数组
  5. 重启jboss出现问题:端口被占用
  6. hibernate-jpa/hibernate-jpa-2.1-api-1.0.0.final.jar源代码下载地址
  7. ipython和python怎么用_如何使用IPython重新加载和自动加载?
  8. SAS宏技术中,%let和call symput有什么区别?
  9. 案例分析:倾斜值传入导致 SQL 资源消耗升高
  10. 官宣预热iQOO 7强悍配置:“性能铁三角”加持 给你强悍全感
  11. Microsoft.Ink namespace
  12. 马云的经典语录(转载)
  13. 一、1.kaggel简街市场预测—baseline代码解析
  14. bat脚本 拷贝文件/文件夹到目标目录
  15. GPS控制网技术设计、技术设计书、作业模式
  16. 使用sklearn构建完整的回归项目(一)
  17. Programiranje
  18. 微信weui之actionSheet应用
  19. 蜗牛机器i211网卡驱动_5个很棒的蜗牛邮件驱动的艺术项目
  20. 微信小程序开发的基础学习

热门文章

  1. Codeforces Round #619 (Div. 2)
  2. 大促系统全流量压测及稳定性保证——京东交易架构分享
  3. Maven: Non-resolvable import POM:Failure to find *** in *** was cached in the local repository.
  4. 自动驾驶“稳打地基”,小鹏汽车基于阿里云建自动驾驶AI智算中心算力可达600PFLOPS
  5. 从计算机屏幕上抓取动态操作过程 也称为,计算机学业水平考试单项选择题综合训练一 答案复习过程...
  6. 长沙麻将APP思路整理
  7. 2014年CCNU-ACM暑期集训总结
  8. 记录一次生产应用启动后发生多次FullGc的解决过程
  9. TIOBE 编程语言排行,各个语言优缺点,以及你适合那种编程语言
  10. 办公室日常管理信息系统 数据库课程设计