当前位置:   article > 正文

音视频开发(三十七):ExoPlayer介绍及简单使用_exoplayer-ui

exoplayer-ui

目录

  1. ExoPlayer基本介绍

  2. ExoPlayer的基本使用

一、ExoPlayer基本介绍

1.1 ExoPlayer优缺点
ExoPlayer是谷歌开源的一个应用级的音视频播放器。ExoPlayer 支持基于 HTTP 的动态自适应流 (DASH)、SmoothStreaming 和通用加密、以及可以很好的支持播放队列、播放源的无缝切换等功能。它采用易于自定义和扩展的设计。
内部的实现也是调用了低层API,比如:MediaCodec、AudioTrack等

画张表格来对比下ExoPlayer和MediaPlayer,更直观的了解

红色框框起来的,核心部分加ui的library也是我们这个系列学习使用重点。

1.2 ExoPlayer架构设计
ExoPlayer的核心是ExoPlayer的接口,其中定义了包含传统播放器的功能(缓冲音视频、播放、暂停、seek等)。ExoPlayer没有设定可以播放的媒体类型、存储方式以及渲染方式,也没有直接实现加载和播放。而是在播放器被创建或者准备播放时将这些工作代理给注册的组件来实现。下面是一些常见ExoPlayer的组件实现:

  1. MediaSource 加载媒体,通过ExoPlayer.prepare注册

  2. TrackSelector:音/视轨提取器,从MediaSource中提取出轨道的数据

  3. Render:对TrackSelector提取出来的数据进行渲染,AudioTrack播放音频、Surface渲染视频

  4. LoadControl:对MediaSource进行控制(什么时候开始缓冲、缓冲多少等)
    ExoPlayer为这些组件提供了默认的实现,如果需要定制可以自定义组件来扩展实现。

通过ExoPlayer的架构图,我们也可以看到其组件模块化的设计,这个架构设计值得学习,也是好的组件/SDK的一个重要要求。在我们的日常项目开发中,开发一个组件 从易用性和以扩展性方面考虑,既要保证使用者很容易上手使用(提供一套默认实现),又要有方便使用者根据自己的场景进行方便的扩展的能力。

本文福利, 免费领取C++音视频学习资料包、技术视频,内容包括(音视频开发,面试题,FFmpeg ,webRTC ,rtmp ,hls ,rtsp ,ffplay ,编解码,推拉流,srs)↓↓↓↓↓↓见下面↓↓文章底部点击免费领取↓↓

1.3 状态机
在看ExoPlayer的状态机之前,我们先看下MeidaPlayer的状态机

可以看到MediaPlayer的状态比较多,使用时如果在不当的位置触发了不匹配的操作,直接会崩溃。
相比MediaPlayer,ExoPlayer的状态少了些,也更容易使用区分,不像MediaPlayer在没有prepared之前都不可以进行播放相关操作,ExoPlayer很多listener以及isplaying的API来进行监听状态变化。ExoPlayer的四种状态如下

  1. /**
  2. * Playback state. One of {@link #STATE_IDLE}, {@link #STATE_BUFFERING}, {@link #STATE_READY} or
  3. * {@link #STATE_ENDED}.
  4. */
  5. @Retention(RetentionPolicy.SOURCE)
  6. @IntDef({STATE_IDLE, STATE_BUFFERING, STATE_READY, STATE_ENDED})
  7. @interface State {}
  8. /** The player does not have any media to play. */
  9. int STATE_IDLE = 1;
  10. /**
  11. * The player is not able to immediately play from its current position. This state typically
  12. * occurs when more data needs to be loaded.
  13. */
  14. int STATE_BUFFERING = 2;
  15. /**
  16. * The player is able to immediately play from its current position. The player will be playing if
  17. * {@link #getPlayWhenReady()} is true, and paused otherwise.
  18. */
  19. int STATE_READY = 3;
  20. /** The player has finished playing the media. */
  21. int STATE_ENDED = 4;

STATE_IDLE:初始状态,此时播放器没有可以播放的资源,播放器停止播放或者播放失败后也会处于该状态
STATE_BUFFERING: 没有足够的数据可以加载播放,此时无法立即播放
STATE_READY : 播放器可以立即播放,是否播放取决于playWhenReady的值,该值表达了使用者的意愿,为true,将会开始播放,否则不播。
STATE_ENDED: 播放完了所有的资源后处于改状态

二、ExoPlayer的简单使用

这一小节我们学习实践ExoPlayer的使用

2.1 AS中引入library
ExoPlayer有很好的扩展性和可定制性,可以根据项目需要进行选择对应的模块,也可以全部包含。

  1. exoplayer-core: Core functionality (required).
  2. exoplayer-dash: Support for DASH content.
  3. exoplayer-hls: Support for HLS content.
  4. exoplayer-smoothstreaming: Support for SmoothStreaming content.
  5. exoplayer-ui: UI components and resources for use with ExoPlayer.

我们根据需要来添加library

  1. implementation 'com.google.android.exoplayer:exoplayer-core:2.13.3'
  2. implementation 'com.google.android.exoplayer:exoplayer-ui: 2.13.3'

接下来出创建一个容器PlayerView以及ExoPlayerView进行播放

2.2 创建播放器、绑定播放器容器、设置数据源、prepare

  1. //1. 创建播放器
  2. player = SimpleExoPlayer.Builder(this).build()
  3. printCurPlaybackState("init") // 此时处于STATE_IDLE = 1;
  4. //2. 播放器和播放器容器绑定
  5. playerView.player = player
  6. //3. 设置数据源
  7. //音频
  8. val mediaItem = MediaItem.fromUri(" https://storage.googleapis.com/exoplayer-test-media-0/play.mp3")
  9. player.setMediaItem(mediaItem)
  10. //4.当Player处于STATE_READY状态时,进行播放
  11. player.playWhenReady = true
  12. //5. 调用prepare开始加载准备数据,该方法时异步方法,不会阻塞ui线程
  13. player.prepare()
  14. printCurPlaybackState("prepare") // 此时处于 STATE_BUFFERING = 2;

2.3 播放监听

当前是否在播放中

  1. public final boolean isPlaying() {
  2. return getPlaybackState() == Player.STATE_READY
  3. && getPlayWhenReady()
  4. && getPlaybackSuppressionReason() == PLAYBACK_SUPPRESSION_REASON_NONE;
  5. }

播放状态改变的listener、音频相关的listener、视频相关的listener

  1. playbackListener = PlaybackListener()
  2. player.addListener(playbackListener)
  3. player.addAudioListener(playbackListener)
  4. player.addVideoListener(playbackListener)
  5. class PlaybackListener : Player.EventListener, AudioListener, VideoListener {
  6. override fun onPlaybackStateChanged(playbackState: Int) {
  7. val stateString: String
  8. stateString = when (playbackState) {
  9. ExoPlayer.STATE_IDLE -> "ExoPlayer.STATE_IDLE -"
  10. ExoPlayer.STATE_BUFFERING -> "ExoPlayer.STATE_BUFFERING -"
  11. ExoPlayer.STATE_READY -> "ExoPlayer.STATE_READY -"
  12. ExoPlayer.STATE_ENDED -> "ExoPlayer.STATE_ENDED -" //播放列表存在时播放最后一个播放完成才会回掉该方法
  13. else -> "UNKNOWN_STATE -"
  14. }
  15. Log.d("ExoBaseUserActivity", "changed state to $stateString")
  16. }
  17. override fun onAudioSessionIdChanged(audioSessionId: Int) {
  18. Log.d("ExoBaseUserActivity", "onAudioSessionIdChanged--sessionId=" + audioSessionId)
  19. }
  20. override fun onAudioAttributesChanged(audioAttributes: AudioAttributes) {
  21. Log.d("ExoBaseUserActivity", "onAudioAttributesChanged--audioAttributes=" + audioAttributes.toString())
  22. }
  23. override fun onVolumeChanged(volume: Float) {
  24. Log.d("ExoBaseUserActivity", "onVolumeChanged--volume=" + volume)
  25. }
  26. override fun onSkipSilenceEnabledChanged(skipSilenceEnabled: Boolean) {
  27. Log.d("ExoBaseUserActivity", "onSkipSilenceEnabledChanged--skipSilenceEnabled=" + skipSilenceEnabled)
  28. }
  29. override fun onVideoSizeChanged(width: Int, height: Int, unappliedRotationDegrees: Int, pixelWidthHeightRatio: Float) {
  30. Log.d("ExoBaseUserActivity", "onVideoSizeChanged--width=" + width + " height=" + height + " unappliedRotationDegrees=" + unappliedRotationDegrees + " pixelWidthHeightRatio=" + pixelWidthHeightRatio)
  31. }
  32. override fun onSurfaceSizeChanged(width: Int, height: Int) {
  33. Log.d("ExoBaseUserActivity", "onSurfaceSizeChanged--width=" + width + " height=" + height)
  34. }
  35. override fun onRenderedFirstFrame() {
  36. Log.d("ExoBaseUserActivity", "onRenderedFirstFrame")
  37. }
  38. }
  39. 用于分析用的listener(会输出更详细的信息)
  40. //通过AnalyticsListener可以输出更多信息
  41. analyticsListener = EventLogger(DefaultTrackSelector())
  42. player.addAnalyticsListener(analyticsListener)
  43. 2.4 释放资源
  44. 在页面不可见/销毁(看是否需要后台播放)时要释放资源
  45. override fun onDestroy() {
  46. super.onDestroy()
  47. player.removeAnalyticsListener(analyticsListener)
  48. player.removeListener(playbackListener)
  49. player.removeAudioListener(playbackListener)
  50. player.removeVideoListener(playbackListener)
  51. player.release()
  52. }

用于分析用的listener(会输出更详细的信息)

  1. //通过AnalyticsListener可以输出更多信息
  2. analyticsListener = EventLogger(DefaultTrackSelector())
  3. player.addAnalyticsListener(analyticsListener)

 2.4 释放资源
在页面不可见/销毁(看是否需要后台播放)时要释放资源

  1. override fun onDestroy() {
  2. super.onDestroy()
  3. player.removeAnalyticsListener(analyticsListener)
  4. player.removeListener(playbackListener)
  5. player.removeAudioListener(playbackListener)
  6. player.removeVideoListener(playbackListener)
  7. player.release()
  8. }

收获

通过本次学习实践收获如下:

  1. 了解ExoPlayer的背景以及相比MediaPlayer的优缺点

  2. 了解ExoPlayer的基本功能

本文福利, 免费领取C++音视频学习资料包、技术视频,内容包括(音视频开发,面试题,FFmpeg ,webRTC ,rtmp ,hls ,rtsp ,ffplay ,编解码,推拉流,srs)↓↓↓↓↓↓见下面↓↓文章底部点击免费领取↓↓

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/繁依Fanyi0/article/detail/299340
推荐阅读
相关标签
  

闽ICP备14008679号