ExoPlayer 使用 Android 的 MediaDrm
API 来支持受 DRM 保护的播放。
下表介绍了不同受支持的 DRM 方案所需的最低 Android 版本,以及支持的流式传输格式:
DRM 方案 | Android 版本号 | Android API 级别 | 支持的格式 |
---|---|---|---|
Widevine“cenc” | 4.4 | 19 | DASH、HLS(仅限 FMP4) |
Widevine“cbcs” | 7.1 | 25 | DASH、HLS(仅限 FMP4) |
ClearKey“cenc” | 5.0 | 21 | DASH |
PlayReady SL2000“cenc” | Android TV | Android TV | DASH、SmoothStreaming、HLS(仅限 FMP4) |
若要使用 ExoPlayer 播放受 DRM 保护的内容,应在构建媒体项时指定 DRM 系统的 UUID 和许可服务器 URI。然后,播放器将使用这些属性构建适用于大多数用例的 DrmSessionManager
的默认实现(名为 DefaultDrmSessionManager
)。对于某些用例,可能还需要其他 DRM 属性,如以下部分所述。
密钥轮替
如需使用轮替键播放流,请在构建媒体项时将 true
传递给 MediaItem.DrmConfiguration.Builder.setMultiSession
。
多键内容
多键内容由多个信息流组成,其中一些信息流使用的键不同。多密钥内容可通过以下两种方式之一播放,具体取决于许可服务器的配置方式。
案例 1:许可服务器做出内容的所有密钥响应
在这种情况下,许可服务器已配置为:在接收到针对一个密钥的请求时,在响应中返回内容的所有密钥。这种情况由 ExoPlayer 处理,无需任何特殊配置。视频流(例如标清和高清视频)之间的无缝自适应,即使它们使用不同的按键,也是如此。
如果可能,我们建议您将许可服务器配置为以这种方式运作。这是支持多键内容播放的最高效且最可靠的方法,因为它不需要客户端发出多个许可请求即可访问不同的信息流。
案例 2:许可服务器仅返回所请求的密钥
在这种情况下,许可服务器配置为仅使用请求中指定的密钥进行响应。只需在构建媒体项时将 true
传递给 MediaItem.DrmConfiguration.Builder.setMultiSession
,便可使用此许可服务器配置播放多键内容。
我们不建议您将许可服务器配置为以这种方式运行。它需要额外的许可请求才能播放多密钥内容,这种方式的效率和稳健性不及上述替代方案。
离线键
您可以在构建媒体项时将键集 ID 传递给 MediaItem.DrmConfiguration.Builder.setKeySetId
,从而加载离线键集。这样便可使用离线键集(其中包含指定 ID)中存储的键进行播放。
针对清除内容的 DRM 会话
使用占位符 DrmSessions
可让 ExoPlayer
针对清除内容使用与播放加密内容时相同的解码器。如果媒体同时包含清晰部分和加密部分,您可能需要使用占位符 DrmSessions
,以避免在清晰部分和加密部分之间发生过渡时重新创建解码器。您可以在构建媒体内容时将 true
传递给 MediaItem.DrmConfiguration.Builder.forceSessionsForAudioAndVideoTracks
,从而允许对音频和视频轨道使用占位符 DrmSessions
。
使用自定义 DrmSessionManager
如果应用想要自定义用于播放的 DrmSessionManager
,则可以实现 DrmSessionManagerProvider
并将其传递给构建播放器时使用的 MediaSource.Factory
。提供程序可以选择是否每次都实例化一个新的管理器实例。如需始终使用相同的实例,请执行以下操作:
Kotlin
val customDrmSessionManager: DrmSessionManager = CustomDrmSessionManager() // Pass a drm session manager provider to the media source factory. val mediaSourceFactory = DefaultMediaSourceFactory(context).setDrmSessionManagerProvider { customDrmSessionManager }
Java
DrmSessionManager customDrmSessionManager = new CustomDrmSessionManager(/* ... */ ); // Pass a drm session manager provider to the media source factory. MediaSource.Factory mediaSourceFactory = new DefaultMediaSourceFactory(context) .setDrmSessionManagerProvider(mediaItem -> customDrmSessionManager);
提高播放性能
在搭载从 Android 6.0(API 级别 23)到 Android 11(API 级别 30)的任何 Android 版本的设备上,播放受 DRM 保护的内容时,如果您遇到视频卡顿,则可以尝试启用异步缓冲区队列。