在安卓设备上播放视频和音乐是很受欢迎的活动。Android框架提供了MediaPlayer作为一个快速的解决方案,可以用最少的代码来播放媒体。Android还提供低级别的媒体api框架,如MediaCodec、AudioTrack和MediaDrm,可用于构建自定义媒体播放器解决方案。
ExoPlayer是一款开源的应用级媒体播放器,基于Android的低级媒体API构建。本指南描述了ExoPlayer库及其使用。它是指ExoPlayer的主要演示应用程序中的代码,以提供具体的示例。该指南介绍了使用ExoPlayer的优缺点。它展示了如何使用ExoPlayer播放DASH,SmoothStreaming和HLS自适应流,以及MP4、M4A、FMP4、WebM、MKV、MP3、Ogg、WAV、MPEG-TS、MPEG-PS、FLV和ADTS (AAC)的格式。它还讨论了ExoPlayer事件、消息、自定义和DRM支持。
ExoPlayer是Android的应用程序级媒体播放器。 它提供了Android的MediaPlayer API的替代品,用于在本地和互联网上播放音频和视频。 ExoPlayer支持Android MediaPlayer API目前不支持的功能,包括DASH和SmoothStreaming自适应回放。 与MediaPlayer API不同,ExoPlayer易于定制和扩展,并可通过Play Store应用程序更新进行更新。
二、优点和缺点
ExoPlayer与MediaPlayer内置的Android相比具有许多优势:
缺点:
三、该库的功能模块概述
ExoPlayer库的核心是ExoPlayer接口。ExoPlayer暴露了普遍使用的高级媒体播放器api功能,比如缓冲媒体、播放、暂停和拖动条的功能。实现的目的是关于对(并因此加以很少的限制)所播放的媒体类型、存储方式和存储方式、以及如何呈现的方式进行很少的假设。ExoPlayer实现不是直接实现媒体的加载和渲染,而是将这项工作委托给创建播放器或准备播放时注入的组件。
所有ExoPlayer实现的常见组件是:
该库为常见用例提供了这些组件的默认实现,更详细的描述请看以下介绍。
ExoPlayer可以使用这些组件,但是如果您不想用ExoPlayer默认的实现方式,也可以使用自定义实现来构建。例如,可以注入自定义LoadControl来更改播放器的缓冲策略,可以在Android设备上将自定义Renderer注入到Android不支持的视频编解码器。
注入组件以实现播放器功能部件的概念存在于整个库中。上面列出的组件的默认实现可以进一步注入组件,所以许多子组件可以被自定义实现单独替换。例如,默认的MediaSource实现需要通过构造函数注入一个或多个DataSource工厂。通过提供自定义工厂,可以从非标准的源或通过不同的网络栈加载数据。
四、开始入门
使用ExoPlayer写一个简单的用例,主要包括以下步骤:
这些步骤在下面更详细地概述。 有关完整示例,请参阅 主应用程序demo中的PlayerActivity。
(1)将ExoPlayer添加为依赖项
入门的第一步是确保您的项目根目录中的build.gradle文件中包含JCenter和Google存储库。
repositories {
jcenter()
google()
}
接下来,在应用程序moule的build.gradle文件中添加一个依赖项。以下内容将为完整的ExoPlayer库添加一个依赖项:
implementation'com.google.android.exoplayer:exoplayer:2.X.X'
其中2.X.X是您的首选版本。或者,您只能依赖您实际需要的库模块。例如,以下内容将添加对Core,DASH和UI库模块的依赖关系,这可能是播放DASH内容的应用程序所需的:
implementation'com.google.android.exoplayer:exoplayer-core:2.X.X'
implementation 'com.google.android.exoplayer:exoplayer-dash:2.X.X'
implementation 'com.google.android.exoplayer:exoplayer-ui:2.X.X'
下面列出了可用的库模块。向完整的ExoPlayer库添加依赖关系等效于单独添加对所有库模块的依赖关系。
除了库模块外,ExoPlayer还有多个扩展模块,它们依赖于外部库来提供附加功能。这些超出了本指南的范围。有关详细信息,请浏览 扩展目录 及其各自的README文件。
(2)创建播放器
您可以使用ExoPlayerFactory创建一个ExoPlayer实例。 ExoPlayerFactory提供了一系列用于创建具有不同级别定制的ExoPlayer实例的方法。 对绝大多数用例而言,应该使用ExoPlayerFactory.newSimpleInstance方法之一。 这些方法返回SimpleExoPlayer,它扩展了ExoPlayer,添加了额外的高级播放器功能。 下面的代码是创建SimpleExoPlayer的示例。
//1.创建默认的 TrackSelector
Handler mainHandler = newHandler();
BandwidthMeter bandwidthMeter = newDefaultBandwidthMeter();
TrackSelection.Factory videoTrackSelectionFactory =
newAdaptiveTrackSelection.Factory(bandwidthMeter);
TrackSelector trackSelector =
newDefaultTrackSelector(videoTrackSelectionFactory);
//2.创建播放器
SimpleExoPlayer player =
ExoPlayerFactory.newSimpleInstance(context, trackSelector);
(3)将播放器附加到view
ExoPlayer库提供了一个PlayerView,它封装了一个PlayerControlView和一个显示视频的Surface。可以将PlayerView包含在应用程序的布局xml中。将播放器绑定到view很简单,代码如下:
// 将播放器附加到view
playerView.setPlayer(player);
如果您需要对播放器控件和渲染视频的Surface进行更详细的控制,则可以分别使用SimpleExoPlayer的setVideoSurfaceView,setVideoTextureView,setVideoSurfaceHolder和setVideoSurface方法直接设置播放器的目标SurfaceView,TextureView,SurfaceHolder或Surface。 您可以将PlayerControlView作为独立组件使用,或者实现您自己的播放控件,直接与播放器交互。 可以使用setTextOutput和setId3Output在播放过程中接收字幕和ID3元数据输出。
(4)准备播放器
在ExoPlayer中,每个media都由MediaSource表示。要播放一段media,您必须先创建一个相应的MediaSource,然后将该对象传递给ExoPlayer.prepare。 ExoPlayer库为DASH(DashMediaSource),SmoothStreaming(SsMediaSource),HLS(HlsMediaSource)和常规媒体文件(ExtractorMediaSource)提供MediaSource实现。 这些实现在本指南后面会有更详细的介绍。
以下代码显示了如何使用适用于播放MP4文件的MediaSource准备播放器。
// 在播放期间测量带宽。 如果不需要,可以为null
DefaultBandwidthMeter bandwidthMeter = newDefaultBandwidthMeter();
// 生成用于加载媒体数据的 DataSource 实例
DataSource.Factory dataSourceFactory = newDefaultDataSourceFactory(context,
Util.getUserAgent(context, "yourApplicationName"), bandwidthMeter);
// 这是要播放媒体的MediaSource
MediaSource videoSource = newExtractorMediaSource.Factory(dataSourceFactory)
.createMediaSource(mp4VideoUri);
// 准备播放器的资源
player.prepare(videoSource);
(5)控制播放器
播放器准备就绪后,可以通过播放器上的调用方法来控制播放。 例如:
setPlayWhenReady可用于开始和暂停播放
各种seekTo方法可用于在媒体内搜索
setRepeatMode可用于控制媒体是否以及如何循环播放
并且setPlaybackParameters可用于调整播放速度和音调。
如果玩家绑定到PlayerView或PlayerControlView,则用户与这些组件的交互将导致玩家调用相应的方法。
(6)释放播放器
当播放器不再需要时释放播放器非常重要,以释放视频解码器等有限资源以供其他应用程序使用。 这可以通过调用ExoPlayer.release完成。
五、媒体资源(MediaSource)的使用
在ExoPlayer中,每个media都由 MediaSource 表示。 ExoPlayer库为DASH(DashMediaSource),SmoothStreaming(SsMediaSource),HLS(HlsMediaSource)和常规媒体文件(ExtractorMediaSource)提供了MediaSource实现。 在 main demo app 的PlayerActivity中可以找到如何实例化所有四个示例。
MediaSource实例不适用于重新使用的情况。 如果您想用相同的media多次准备播放器,请每次使用新的实例。
除了上述的MediaSource实现外,ExoPlayer库还提供了MergingMediaSource,LoopingMediaSource,ConcatenatingMediaSource和DynamicConcatenatingMediaSource。这些MediaSource实现可以通过组合来实现更复杂的播放功能。下面描述了一些常见的用例。请注意,尽管在视频播放的上下文中描述了以下示例,但它们同样适用于仅播放音频,以及任何支持的媒体类型的播放的情况。
(1)从侧面加载字幕文件
给定一个视频文件和一个单独的字幕文件,可以使用 MergingMediaSource将它们合并到单个播放源中。
// Build the video MediaSource.
MediaSource videoSource =
newExtractorMediaSource.Factory(...).createMediaSource(videoUri);
// Build the subtitle MediaSource.
Format subtitleFormat = Format.createTextSampleFormat(
id, // An identifier for the track. May be null.
MimeTypes.APPLICATION_SUBRIP, // The mime type. Must be set correctly.
selectionFlags, // Selection flags for the track.
language); // The subtitle language. May be null.
MediaSource subtitleSource =
newSingleSampleMediaSource.Factory(...)
.createMediaSource(subtitleUri, subtitleFormat, C.TIME_UNSET);
// Plays the video with the sideloaded subtitle.
MergingMediaSource mergedSource =
newMergingMediaSource(videoSource, subtitleSource);
(2)循环播放视频
要无限循环,通常最好使用 ExoPlayer.setRepeatMode而不是 LoopingMediaSource。
使用 LoopingMediaSource可以将视频无缝地循环固定次数。 以下是播放视频两次的示例。
MediaSource source =
newExtractorMediaSource.Factory(...).createMediaSource(videoUri);
// Plays the video twice.
LoopingMediaSource loopingSource = newLoopingMediaSource(source, 2);
(3)播放一系列视频
ConcatenatingMediaSource可以连续播放两个或多个单独的MediaSource。 下面是按顺序播放了两个视频的例子。 数据源之间的转换是无缝的。对连接的源具有相同的格式这一点不做强制要求,您可以把两个不同格式的数据源连接起来(例如,将包含480p H264的视频文件与包含720p VP9的视频文件连接起来就可以)。 同时这些源甚至可以是不同类型的(例如,将视频与仅音频流串接也是很友好的)。
MediaSource firstSource =
newExtractorMediaSource.Factory(...).createMediaSource(firstVideoUri);
MediaSource secondSource =
newExtractorMediaSource.Factory(...).createMediaSource(secondVideoUri);
// Plays the first video, then the second video.
ConcatenatingMediaSource concatenatedSource =
newConcatenatingMediaSource(firstSource, secondSource);
DynamicConcatenatingMediaSource类似于 ConcatenatingMediaSource,不同之处在于它允许在播放前和播放期间动态添加,删除和移动MediaSource。 DynamicConcatenatingMediaSource非常适合于播放列表的使用场景,即用户可以在播放期间修改播放列表。
MediaSource 实例不应该多次添加到 DynamicConcatenatingMediaSource中,或者在之前被删除的情况下重新添加。 推荐创建新的实例去操作。
(4)高级组合
有可能进一步将复合MediaSources组合起来,用于更多不常见的用法。 给定两个视频A和B,以下示例显示LoopingMediaSource和ConcatenatingMediaSource如何一起使用来播放序列(A,A,B)。
MediaSource firstSource =
newExtractorMediaSource.Factory(...).createMediaSource(firstVideoUri);
MediaSource secondSource =
newExtractorMediaSource.Factory(...).createMediaSource(secondVideoUri);
// Plays the first video twice.
LoopingMediaSource firstSourceTwice = newLoopingMediaSource(firstSource, 2);
// Plays the first video twice, then the second video.
ConcatenatingMediaSource concatenatedSource =
newConcatenatingMediaSource(firstSourceTwice, secondSource);
下面的例子和上面那个示例实质是一样的,表明可以有多种方式来实现相同的结果。
MediaSource firstSource = newExtractorMediaSource.Builder(firstVideoUri, ...).build();
MediaSource secondSource = newExtractorMediaSource.Builder(secondVideoUri, ...).build();
// Plays the first video twice, then the second video.
ConcatenatingMediaSource concatenatedSource =
newConcatenatingMediaSource(firstSource, firstSource, secondSource);
除非根据文档明确允许,否则避免在组合中多次使用相同的MediaSource实例很重要。 上面的示例中使用firstSource两次就是这种情况,因为用于ConcatenatingMediaSource的Javadoc明确指出允许重复条目。 然而,一般来说,由构造组成的对象的图形应该是树形结构(这个地方不好翻译,英文不好见谅)。 在组合中使用多个等效的MediaSource实例是允许的。
六、播放器事件
在播放过程中,您的应用程序可以侦听由ExoPlayer生成的 显示播放器整体状态的事件。 这些事件对于更新用户界面组件(如播放控件)非常有用。 许多ExoPlayer组件还会报告它们自己组件特定的低级别事件,这对性能监视非常有用。
(1)高等级事件
ExoPlayer允许通过调用 addListener 和 removeListener 方法来添加和删除 EventListener。 已注册的监听器会收到播放状态更改以及何时发生导致播放失败的错误的通知。
实现自定义播放控制的开发人员应该注册一个监听器,并在播放器的状态发生变化时使用它来更新控件。 如果播放失败,应用程序还应该向用户显示适当的错误信息。
使用SimpleExoPlayer时,可以在播放器上设置其他监听器。 特别要说明的是,addVideoListener 允许应用程序接收可能对调整UI有用的视频呈现相关的事件(例如,正在呈现视频的 Surface 的高宽比)。 监听器也可以设置为接收调试信息,例如调用 setVideoDebugListener 和 setAudioDebugListener。
(2)低等级事件
除了高级监听器之外,ExoPlayer库提供的许多单独组件允许自己的事件监听器。 通常需要将 Handler 对象传递给这些组件,这决定了调用监听器方法的线程。 在大多数情况下,您应该使用 Handler 与app的主线程关联。
七、将消息发送到组件
可以将消息发送到ExoPlayer组件,可以使用createMessage创建,然后使用PlayerMessage.send发送。 默认情况下,消息尽快在回放线程上传递,但可以通过设置另一个回调线程(使用PlayerMessage.setHandler)或通过指定传递播放位置(使用PlayerMessage.setPosition)来定制消息。 通过ExoPlayer发送消息可确保操作按照播放器上正在执行的任何其他操作的顺序执行。
大多数ExoPlayer的开箱即用的渲染器支持在回放期间允许对其配置进行更改的消息。 例如,音频渲染器接受消息来设置音量,视频渲染器接受消息来设置Surface。 这些消息应该在回放线程上传递以确保线程安全
八、定制
ExoPlayer相比Android的MediaPlayer的主要优点之一是可以自定义和扩展播放器,以更好地适应开发人员的使用情况。 ExoPlayer库专为此设计的,定义了许多接口和抽象基类,使应用程序开发人员可以轻松地替换库提供的默认实现。
(1)构建自定义组件的一些示范:
(2)定制指南
九、数字版权管理
在Android 4.4 (API级别19)和更高版本中,ExoPlayer支持数字版权管理(DRM)保护回放。
为了使用ExoPlayer播放DRM保护的内容,您的应用程序必须在实例化播放器时注入DrmSessionManager。ExoPlayerFactory提供了允许这种情况的工厂方法。DrmSessionManager对象负责提供DrmSession实例,该实例为解密提供了MediaCrypto对象,并确保所需的解密密钥可用于正在使用的底层DRM模块。
ExoPlayer库提供了一个DrmSessionManager的默认实现,名为DefaultDrmSessionManager,它使用MediaDrm。会话管理器支持在设备上存在模块DRM组件的任何DRM方案。所有的Android设备都需要支持Widevine模块DRM(使用L3安全性,尽管许多设备也支持L1)。某些设备可能支持其他方案,例如PlayReady。所有的Android TV 设备都支持PlayReady。
主演示应用程序 中的PlayerActivity演示了如何在实例化播放器时创建和注入DefaultDrmSessionManager。
ExoPlayer地址:https://github.com/google/ExoPlayer