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,还可以提供其他属性。然后,播放器将使用这些属性构建适用于大多数用例的 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
,以避免在明文和加密部分之间发生转换时重新创建解码器。如需为音频和视频轨道使用占位符 DrmSessions
,只需在构建媒体项时将 true
传递给 MediaItem.DrmConfiguration.Builder.forceSessionsForAudioAndVideoTracks
即可。
使用自定义 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 保护的内容时遇到视频卡顿问题,可以尝试启用异步缓冲区队列。