Android之播放器的恢复

在Android中 一个简单的播放器只需要MediaPlayer和SurfaceView就能完成,当然,若需要更多的功能,比如快进、快退都是可以的,只是暂时讨论的是最简单的。

播放器有好多种状态,正如下面的这张图所示,这个MediaPlayer的状态转换图也表征了它的生命周期,搞清楚这个图后,在使用MediaPlayer时,考虑情况将会更周全,写出的代码也更具健壮性有。
这里写图片描述
这张状态转换图清晰的描述了MediaPlayer的各个状态,也列举了主要的方法的调用时序,每种方法只能在一些特定的状态下使用,如果使用时MediaPlayer的状态不正确则会引发IllegalStateException异常。对于这张图的解读,后面有,这儿就暂时不描述了,先来看这样一个问题。
简单地,写一个播放器:

public class MainActivity extends Activity implements MediaPlayer.OnPreparedListener{
    private String videoUrl = "http://172.16.10.179/mu_fu_feng_yun_1.mp4";
    private MediaPlayer mediaPlay;


    protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    mediaPlay = new MediaPlayer();
    try {
        mediaPlay.setDataSource(videoUrl);
    } catch (IllegalArgumentException e) {
        e.printStackTrace();
    } catch (SecurityException e) {
        e.printStackTrace();
    } catch (IllegalStateException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }
    mediaPlay.prepareAsync();
    mediaPlay.setOnPreparedListener(this);
}
    @Override
    protected void onResume() {
        super.onResume();
    }
    @Override
    protected void onStop() {
        super.onStop();
    }
    @Override
    public void onPrepared(MediaPlayer mp) {
        mediaPlay.start();
    }

    @Override
    protected void onDestroy() {
        mediaPlay.release();
        super.onDestroy();
    }
}

这段程序可以播放视频吗?可以,只是有声音没有视频,因为没有给它设置显示器(surfaceView)。
来看一下做的工作:

  • 初始化mediaplayer
  • 设置播放路径
  • 异步准备
  • 设置准备好之后的监听,在准备好之后就播放

那么,现在想给它加一个显示器,应该怎么加?(下面的这个代码有错误)

public class MainActivity extends Activity implements MediaPlayer.OnPreparedListener{
    private SurfaceView surfaceView;
    private String videoUrl = "http://172.16.10.179/mu_fu_feng_yun_1.mp4";
    private MediaPlayer mediaPlay;


    protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    surfaceView = (SurfaceView) findViewById(R.id.mediaVideoSurface);
    surfaceView.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
    surfaceView.setKeepScreenOn(true);

    mediaPlay = new MediaPlayer();
    mediaPlay.setDisplay(surfaceView.getHolder());
    try {
        mediaPlay.setDataSource(videoUrl);
    } catch (IllegalArgumentException e) {
        e.printStackTrace();
    } catch (SecurityException e) {
        e.printStackTrace();
    } catch (IllegalStateException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }
    mediaPlay.prepareAsync();
    mediaPlay.setOnPreparedListener(this);
}
    @Override
    protected void onResume() {
        super.onResume();
    }
    @Override
    protected void onStop() {
        super.onStop();
    }
    @Override
    public void onPrepared(MediaPlayer mp) {
        mediaPlay.start();
    }

    @Override
    protected void onDestroy() {
        mediaPlay.release();
        super.onDestroy();
    }
}

这样写对吗?一运行,果断崩溃。看起来挺合理的啊,在mediaPlayer创建之前,创建了surfaceView,创建之后,设置显示,但是却是错误。

java.lang.IllegalArgumentException: The surface has been released

好吧,表面已经被释放。。。怎么已经被释放了,应该是还没创建好吧。
那么这样写试试:

public class MainActivity extends Activity implements MediaPlayer.OnPreparedListener{
    private SurfaceView surfaceView;
    private String videoUrl = "http://172.16.10.179/mu_fu_feng_yun_1.mp4";
    private MediaPlayer mediaPlay;
    private String TAG = MainActivity.class.getSimpleName();

    protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    surfaceView = (SurfaceView) findViewById(R.id.mediaVideoSurface);
    surfaceView.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
    surfaceView.setKeepScreenOn(true);

    mediaPlay = new MediaPlayer();
    try {
        mediaPlay.setDataSource(videoUrl);
    } catch (IllegalArgumentException e) {
        e.printStackTrace();
    } catch (SecurityException e) {
        e.printStackTrace();
    } catch (IllegalStateException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }
    mediaPlay.prepareAsync();
    mediaPlay.setOnPreparedListener(this);
}
    @Override
    protected void onResume() {
        super.onResume();
        Log.d(TAG, "run onResume");
    }
    @Override
    protected void onStop() {
        super.onStop();
    }
    @Override
    public void onPrepared(MediaPlayer mp) {
        mediaPlay.start();
        mediaPlay.setDisplay(surfaceView.getHolder());
        Log.d(TAG, "run setDisplay");
    }

    @Override
    protected void onDestroy() {
        mediaPlay.release();
        super.onDestroy();
    }
}

将mediaPlay.setDisplay(surfaceView.getHolder());移到了mediaPlay.start();之后,并且添加了onResume中的log打印和setDisplay的log打印。
这里写图片描述
从图中可以看到,在界面创建完成之后,才setDisplay,这时是可以正常播放的。所以这个setDisplay一定要在surfaceView创建完成之后才能设置,具体怎么设置,继续看。

但是,有个问题,当按下home键,或者干其他的事,发生了activity的stop,返回之后有声音,没有视频。这是怎么回事?

原来SurfaceView它是有创建和销毁的,看下面的代码。

public class MainActivity extends Activity implements MediaPlayer.OnPreparedListener{
    private SurfaceView surfaceView;
    private String videoUrl = "http://172.16.10.179/mu_fu_feng_yun_1.mp4";
    private MediaPlayer mediaPlay;
    private String TAG = MainActivity.class.getSimpleName();

    protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    surfaceView = (SurfaceView) findViewById(R.id.mediaVideoSurface);
    surfaceView.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
    surfaceView.setKeepScreenOn(true);
    surfaceView.getHolder().addCallback(holderCallback);
    mediaPlay = new MediaPlayer();
    try {
        mediaPlay.setDataSource(videoUrl);
    } catch (IllegalArgumentException e) {
        e.printStackTrace();
    } catch (SecurityException e) {
        e.printStackTrace();
    } catch (IllegalStateException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }
    mediaPlay.prepareAsync();
    mediaPlay.setOnPreparedListener(this);
}
    @Override
    protected void onResume() {
        super.onResume();
        Log.d(TAG, "run onResume");
    }
    @Override
    protected void onStop() {
        super.onStop();
    }
    @Override
    public void onPrepared(MediaPlayer mp) {
        mediaPlay.start();
        mediaPlay.setDisplay(surfaceView.getHolder());
        Log.d(TAG, "run setDisplay");
    }

    @Override
    protected void onDestroy() {
        mediaPlay.release();
        super.onDestroy();
    }
    private SurfaceHolder.Callback holderCallback = new SurfaceHolder.Callback() {

        @Override
        public void surfaceCreated(SurfaceHolder holder) {
            Log.d(TAG, "SurfaceHolder Created");
        }

        @Override
        public void surfaceChanged(SurfaceHolder holder, int format, int width,
                int height) {
            Log.d(TAG, "SurfaceHolder Changed");
        }

        @Override
        public void surfaceDestroyed(SurfaceHolder holder) {
            Log.d(TAG, "SurfaceHolder Destroyed");
        }

    };
}

在原来的基础上,给surfaceView的holder添加了一个callback,当程序执行后,会播放视频,然后按下home键,结果如下。
这里写图片描述

可以看到,surfaceView的holder创建了两次,当按下home键后把第一次的会销毁,第二次打开又会创建,但已经不是原来的holder了,当然不会显示了。

将 mediaPlay.setDisplay(surfaceView.getHolder());写到SurfaceHolder.Callback的surfaceCreated中试试,结果可以想像的到,第二次打开能播放了。

 private SurfaceHolder.Callback holderCallback = new SurfaceHolder.Callback() {

        @Override
        public void surfaceCreated(SurfaceHolder holder) {
            Log.d(TAG, "SurfaceHolder Created");
            if(mediaPlay!=null){
                mediaPlay.setDisplay(surfaceView.getHolder());
                Log.d(TAG, "run setDisplay");
            }
        }

        @Override
        public void surfaceChanged(SurfaceHolder holder, int format, int width,
                int height) {
            Log.d(TAG, "SurfaceHolder Changed");
        }

        @Override
        public void surfaceDestroyed(SurfaceHolder holder) {
            Log.d(TAG, "SurfaceHolder Destroyed");
        }

    };

但是问题又来了,在按下home键之后,怎么可以视频还播放呢?好吧,让它暂停吧,那么暂停了,什么时候暂停呢?重新打开,继续播放,什么时候播放?
可以在activity pause的时候暂停,在activity restart的时候让它重新播放。

下面是对mediaplayer各状态的解释,是摘抄过来的,来源已经找不到了,若有侵权,请联系我。

Idle 状态:当使用new()方法创建一个MediaPlayer对象或者调用了其reset()方法时,该MediaPlayer对象处于idle状态。这两种方法的一个重要差别就是:如果在这个状态下调用了getDuration()等方法(相当于调用时机不正确),通过reset()方法进入idle状态的话会触发OnErrorListener.onError(),并且MediaPlayer会进入Error状态;如果是新创建的MediaPlayer对象,则并不会触发onError(),也不会进入Error状态。

End 状态:通过release()方法可以进入End状态,只要MediaPlayer对象不再被使用,就应当尽快将其通过release()方法释放掉,以释放相关的软硬件组件资源,这其中有些资源是只有一份的(相当于临界资源)。如果MediaPlayer对象进入了End状态,则不会在进入任何其他状态了。

Initialized 状态:这个状态比较简单,MediaPlayer调用setDataSource()方法就进入Initialized状态,表示此时要播放的文件已经设置好了。

Prepared 状态:初始化完成之后还需要通过调用prepare()或prepareAsync()方法,这两个方法一个是同步的一个是异步的,只有进入Prepared状态,才表明MediaPlayer到目前为止都没有错误,可以进行文件播放。

Preparing 状态:这个状态比较好理解,主要是和prepareAsync()配合,如果异步准备完成,会触发OnPreparedListener.onPrepared(),进而进入Prepared状态。

Started 状态:显然,MediaPlayer一旦准备好,就可以调用start()方法,这样MediaPlayer就处于Started状态,这表明MediaPlayer正在播放文件过程中。可以使用isPlaying()测试MediaPlayer是否处于了Started状态。如果播放完毕,而又设置了循环播放,则MediaPlayer仍然会处于Started状态,类似的,如果在该状态下MediaPlayer调用了seekTo()或者start()方法均可以让MediaPlayer停留在Started状态。

Paused 状态:Started状态下MediaPlayer调用pause()方法可以暂停MediaPlayer,从而进入Paused状态,MediaPlayer暂停后再次调用start()则可以继续MediaPlayer的播放,转到Started状态,暂停状态时可以调用seekTo()方法,这是不会改变状态的。

Stop 状态:Started或者Paused状态下均可调用stop()停止MediaPlayer,而处于Stop状态的MediaPlayer要想重新播放,需要通过prepareAsync()和prepare()回到先前的Prepared状态重新开始才可以。

PlaybackCompleted状态:文件正常播放完毕,而又没有设置循环播放的话就进入该状态,并会触发OnCompletionListener的onCompletion()方法。此时可以调用start()方法重新从头播放文件,也可以stop()停止MediaPlayer,或者也可以seekTo()来重新定位播放位置。

Error状态:如果由于某种原因MediaPlayer出现了错误,会触发OnErrorListener.onError()事件,此时MediaPlayer即进入Error状态,及时捕捉并妥善处理这些错误是很重要的,可以帮助我们及时释放相关的软硬件资源,也可以改善用户体验。通过setOnErrorListener(android.media.MediaPlayer.OnErrorListener)可以设置该监听器。如果MediaPlayer进入了Error状态,可以通过调用reset()来恢复,使得MediaPlayer重新返回到Idle状态。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android上构建一个视频播放器的架构可以有多种方式。一个常见的方法是使用MediaPlayer类来处理视频的播放和控制。您可以在Activity或Fragment中创建一个MediaPlayer对象,并使用SurfaceView来显示视频内容。当锁屏时,您可以在Activity或Fragment的生命周期方法中处理MediaPlayer的暂停和恢复。例如,在onPause方法中暂停播放器,在onResume方法中恢复播放器。 另一种方法是使用ExoPlayer库,它是Google开发的一个强大的、灵活的媒体播放器框架。ExoPlayer提供了更高级的功能,如自适应流媒体、DRM保护、字幕和广告插入等。您可以在ExoPlayer官方文档中找到更多关于如何在Android上构建一个视频播放器的详细指南。 总的来说,构建一个Android视频播放器的架构需要考虑到音频和视频数据的播放、界面的展示和控制、生命周期的管理等方面。根据您的需求和技术要求,选择合适的播放器库或自定义实现来构建您的视频播放器架构。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [linux开源视频播放器_3个适用于Linux的开源音乐播放器](https://blog.csdn.net/cumj63710/article/details/107391565)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* [Android有关surfaceView重新创建的问题。](https://blog.csdn.net/a1010012805/article/details/46738581)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值