记录一下,没啥干货,查阅方便

状态图详解

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

MediaRecorder 设置方向_sed

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状态
 

使用方法举例:

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;
     }    @Override
     public void onPrepared(MediaPlayer mp) {
         hasPrepared = true; // 准备完成后回调到这里
         start();
     }    @Override
     public void onCompletion(MediaPlayer mp) {
         hasPrepared = false;
         // 通知调用处,调用play()方法进行下一个曲目的播放
     }    @Override
     public boolean onError(MediaPlayer mp, int what, int extra) {
         hasPrepared = false;
         return false;
     }
 }