自定义

ExoPlayer 库的核心是 Player 接口。一个 Player 提供了传统的媒体播放器的高级功能,例如缓冲媒体、播放、暂停和跳转。默认实现 ExoPlayer 旨在对正在播放的媒体类型、存储方式和位置以及渲染方式做出很少的假设(因此施加很少的限制)。ExoPlayer 实现不会直接实现媒体的加载和渲染,而是将此工作委托给在创建播放器时或将新的媒体源传递给播放器时注入的组件。所有 ExoPlayer 实现共有的组件包括:

  • MediaSource 实例,用于定义要播放的媒体、加载媒体以及从中读取已加载媒体。MediaSource 实例由播放器内部的 MediaSource.FactoryMediaItem 创建。它们也可以使用 基于媒体源的播放列表 API 直接传递给播放器。
  • MediaSource.Factory 实例,用于将 MediaItem 转换为 MediaSourceMediaSource.Factory 在创建播放器时注入。
  • Renderer 实例,用于渲染媒体的各个组成部分。这些在创建播放器时注入。
  • TrackSelector,用于选择 MediaSource 提供的、由每个可用的 Renderer 使用的轨道。TrackSelector 在创建播放器时注入。
  • LoadControl,用于控制 MediaSource 何时缓冲更多媒体以及缓冲多少媒体。LoadControl 在创建播放器时注入。
  • LivePlaybackSpeedControl,用于控制直播播放期间的播放速度,以便播放器能够保持接近配置的直播偏移量。LivePlaybackSpeedControl 在创建播放器时注入。

注入实现播放器功能部分的组件的概念贯穿整个库。某些组件的默认实现将工作委托给进一步注入的组件。这允许许多子组件被单独替换为以自定义方式配置的实现。

播放器自定义

下面描述了一些通过注入组件来自定义播放器的常见示例。

配置网络栈

我们有一页关于 自定义 ExoPlayer 使用的网络栈 的内容。

缓存从网络加载的数据

请参阅有关 临时动态缓存下载媒体 的指南。

自定义服务器交互

某些应用可能需要拦截 HTTP 请求和响应。您可能需要注入自定义请求头、读取服务器的响应头、修改请求的 URI 等。例如,您的应用可以通过在请求媒体片段时将令牌作为头注入来进行身份验证。

以下示例演示了如何通过将自定义 DataSource.Factory 注入 DefaultMediaSourceFactory 来实现这些行为。

Kotlin

val dataSourceFactory =
  DataSource.Factory {
    val dataSource = httpDataSourceFactory.createDataSource()
    // Set a custom authentication request header.
    dataSource.setRequestProperty("Header", "Value")
    dataSource
  }
val player =
  ExoPlayer.Builder(context)
    .setMediaSourceFactory(
      DefaultMediaSourceFactory(context).setDataSourceFactory(dataSourceFactory)
    )
    .build()

Java

DataSource.Factory dataSourceFactory =
    () -> {
      HttpDataSource dataSource = httpDataSourceFactory.createDataSource();
      // Set a custom authentication request header.
      dataSource.setRequestProperty("Header", "Value");
      return dataSource;
    };

ExoPlayer player =
    new ExoPlayer.Builder(context)
        .setMediaSourceFactory(
            new DefaultMediaSourceFactory(context).setDataSourceFactory(dataSourceFactory))
        .build();

在上边的代码片段中,注入的 HttpDataSource 在每个 HTTP 请求中都包含头 "Header: Value"。此行为对于与 HTTP 源的每次交互都是固定的

对于更细粒度的做法,您可以使用 ResolvingDataSource 注入即时行为。以下代码片段显示了如何在与 HTTP 源交互之前注入请求头。

Kotlin

val dataSourceFactory: DataSource.Factory =
  ResolvingDataSource.Factory(httpDataSourceFactory) { dataSpec: DataSpec ->
    // Provide just-in-time request headers.
    dataSpec.withRequestHeaders(getCustomHeaders(dataSpec.uri))
  }

Java

    DataSource.Factory dataSourceFactory =
        new ResolvingDataSource.Factory(
            httpDataSourceFactory,
            // Provide just-in-time request headers.
            dataSpec -> dataSpec.withRequestHeaders(getCustomHeaders(dataSpec.uri)));

您还可以使用 ResolvingDataSource 对 URI 执行即时修改,如下面的代码片段所示。

Kotlin

val dataSourceFactory: DataSource.Factory =
  ResolvingDataSource.Factory(httpDataSourceFactory) { dataSpec: DataSpec ->
    // Provide just-in-time URI resolution logic.
    dataSpec.withUri(resolveUri(dataSpec.uri))
  }

Java

DataSource.Factory dataSourceFactory =
    new ResolvingDataSource.Factory(
        httpDataSourceFactory,
        // Provide just-in-time URI resolution logic.
        dataSpec -> dataSpec.withUri(resolveUri(dataSpec.uri)));

自定义错误处理

实现自定义的 LoadErrorHandlingPolicy 允许应用自定义 ExoPlayer 对加载错误的反应方式。例如,应用可能希望快速失败而不是多次重试,或者可能希望自定义控制播放器在每次重试之间等待时间的回退逻辑。以下代码片段显示了如何实现自定义回退逻辑。

Kotlin

val loadErrorHandlingPolicy: LoadErrorHandlingPolicy =
  object : DefaultLoadErrorHandlingPolicy() {
    override fun getRetryDelayMsFor(loadErrorInfo: LoadErrorInfo): Long {
      // Implement custom back-off logic here.
      return 0
    }
  }
val player =
  ExoPlayer.Builder(context)
    .setMediaSourceFactory(
      DefaultMediaSourceFactory(context).setLoadErrorHandlingPolicy(loadErrorHandlingPolicy)
    )
    .build()

Java

LoadErrorHandlingPolicy loadErrorHandlingPolicy =
    new DefaultLoadErrorHandlingPolicy() {
      @Override
      public long getRetryDelayMsFor(LoadErrorInfo loadErrorInfo) {
        // Implement custom back-off logic here.
        return 0;
      }
    };

ExoPlayer player =
    new ExoPlayer.Builder(context)
        .setMediaSourceFactory(
            new DefaultMediaSourceFactory(context)
                .setLoadErrorHandlingPolicy(loadErrorHandlingPolicy))
        .build();

LoadErrorInfo 参数包含有关失败加载的更多信息,以便根据错误类型或失败的请求自定义逻辑。

自定义提取器标志

提取器标志可用于自定义如何从渐进式媒体中提取各个格式。它们可以设置在提供给 DefaultMediaSourceFactoryDefaultExtractorsFactory 上。以下示例传递一个标志,该标志启用 MP3 流的基于索引的跳转。

Kotlin

val extractorsFactory =
  DefaultExtractorsFactory().setMp3ExtractorFlags(Mp3Extractor.FLAG_ENABLE_INDEX_SEEKING)
val player =
  ExoPlayer.Builder(context)
    .setMediaSourceFactory(DefaultMediaSourceFactory(context, extractorsFactory))
    .build()

Java

DefaultExtractorsFactory extractorsFactory =
    new DefaultExtractorsFactory().setMp3ExtractorFlags(Mp3Extractor.FLAG_ENABLE_INDEX_SEEKING);

ExoPlayer player =
    new ExoPlayer.Builder(context)
        .setMediaSourceFactory(new DefaultMediaSourceFactory(context, extractorsFactory))
        .build();

启用恒定比特率跳转

对于 MP3、ADTS 和 AMR 流,您可以使用 FLAG_ENABLE_CONSTANT_BITRATE_SEEKING 标志启用使用恒定比特率假设的近似跳转。可以使用各个 DefaultExtractorsFactory.setXyzExtractorFlags 方法为各个提取器设置这些标志,如上所述。要为所有支持它的提取器启用恒定比特率跳转,请使用 DefaultExtractorsFactory.setConstantBitrateSeekingEnabled

Kotlin

val extractorsFactory = DefaultExtractorsFactory().setConstantBitrateSeekingEnabled(true)

Java

DefaultExtractorsFactory extractorsFactory =
    new DefaultExtractorsFactory().setConstantBitrateSeekingEnabled(true);

ExtractorsFactory 然后可以通过 DefaultMediaSourceFactory 注入,如上所述自定义提取器标志。

启用异步缓冲区排队

异步缓冲区排队是 ExoPlayer 渲染管道中的一个增强功能,它以 异步模式 操作 MediaCodec 实例,并使用额外的线程来调度数据的解码和渲染。启用它可以减少丢帧和音频欠载。

在运行 Android 12(API 级别 31)及更高版本的设备上,默认情况下启用异步缓冲区排队,并且从 Android 6.0(API 级别 23)开始可以手动启用。考虑为在其中观察到丢帧或音频欠载的特定设备启用此功能,尤其是在播放受 DRM 保护或高帧率内容时。

在最简单的情况下,您需要按如下所示将 DefaultRenderersFactory 注入播放器。

Kotlin

val renderersFactory = 
  DefaultRenderersFactory(context).forceEnableMediaCodecAsynchronousQueueing()
val exoPlayer = ExoPlayer.Builder(context, renderersFactory).build()

Java

DefaultRenderersFactory renderersFactory =
    new DefaultRenderersFactory(context).forceEnableMediaCodecAsynchronousQueueing();
ExoPlayer exoPlayer = new ExoPlayer.Builder(context, renderersFactory).build();

如果直接实例化渲染器,请将 AsynchronousMediaCodecAdapter.Factory 传递给 MediaCodecVideoRendererMediaCodecAudioRenderer 构造函数。

使用 ForwardingPlayer 拦截方法调用

您可以通过将 Player 实例包装在 ForwardingPlayer 的子类中并覆盖方法来自定义 Player 实例的一些行为,以便执行以下任何操作。

  • 在将参数传递给委托 Player 之前访问它们。
  • 在返回之前访问委托 Player 的返回值。
  • 完全重新实现该方法。

覆盖 ForwardingPlayer 方法时,务必确保实现保持一致并符合 Player 接口,尤其是在处理旨在具有相同或相关行为的方法时。例如:

  • 如果要覆盖每个“播放”操作,则需要覆盖 ForwardingPlayer.playForwardingPlayer.setPlayWhenReady,因为调用者期望当 playWhenReady = true 时,这些方法的行为相同。
  • 如果要更改向前跳转增量,则需要覆盖 ForwardingPlayer.seekForward 以使用自定义增量执行跳转,并覆盖 ForwardingPlayer.getSeekForwardIncrement 以将正确的自定义增量报告回调用方。
  • 如果要控制由播放器实例宣传的 Player.Commands,则必须覆盖 Player.getAvailableCommands()Player.isCommandAvailable(),并且还要监听 Player.Listener.onAvailableCommandsChanged() 回调以获取来自底层播放器的更改通知。

MediaSource 自定义

上面的示例注入自定义组件,用于播放传递给播放器的所有 MediaItem 对象。如果需要细粒度的自定义,也可以将自定义组件注入各个 MediaSource 实例,这些实例可以直接传递给播放器。以下示例显示了如何自定义 ProgressiveMediaSource 以使用自定义的 DataSource.FactoryExtractorsFactoryLoadErrorHandlingPolicy

Kotlin

val mediaSource =
  ProgressiveMediaSource.Factory(customDataSourceFactory, customExtractorsFactory)
    .setLoadErrorHandlingPolicy(customLoadErrorHandlingPolicy)
    .createMediaSource(MediaItem.fromUri(streamUri))

Java

ProgressiveMediaSource mediaSource =
    new ProgressiveMediaSource.Factory(customDataSourceFactory, customExtractorsFactory)
        .setLoadErrorHandlingPolicy(customLoadErrorHandlingPolicy)
        .createMediaSource(MediaItem.fromUri(streamUri));

创建自定义组件

该库为常见用例提供了此页顶部列出的组件的默认实现。ExoPlayer 可以使用这些组件,但如果需要非标准行为,也可以构建为使用自定义实现。自定义实现的一些用例包括:

  • Renderer – 您可能需要实现自定义 Renderer 来处理库提供的默认实现不支持的媒体类型。
  • TrackSelector – 实现自定义 TrackSelector 允许应用开发者更改 MediaSource 公开的轨道如何被每个可用的 Renderer 使用的方式。
  • LoadControl – 实现自定义 LoadControl 允许应用开发者更改播放器的缓冲策略。
  • Extractor – 如果需要支持库当前不支持的容器格式,请考虑实现自定义 Extractor 类。
  • MediaSource – 如果希望以自定义方式获取要馈送到渲染器的媒体样本,或者如果希望实现自定义 MediaSource 组合行为,则实现自定义 MediaSource 类可能很合适。
  • MediaSource.Factory – 实现自定义 MediaSource.Factory 允许应用程序自定义从 MediaItem 创建 MediaSource 的方式。
  • DataSource – ExoPlayer 的上游包已经包含许多针对不同用例的 DataSource 实现。您可能希望实现自己的 DataSource 类以通过其他方式加载数据,例如通过自定义协议、使用自定义 HTTP 堆栈或从自定义持久性缓存加载数据。

构建自定义组件时,我们建议:

  • 如果自定义组件需要向应用报告事件,我们建议您使用与现有 ExoPlayer 组件相同的模型来执行此操作,例如使用 EventDispatcher 类或将 Handler 与侦听器一起传递到组件的构造函数。
  • 我们建议自定义组件使用与现有 ExoPlayer 组件相同的模型,以便应用程序在播放过程中可以重新配置它们。为此,自定义组件应实现PlayerMessage.Target并在handleMessage方法中接收配置更改。应用程序代码应通过调用 ExoPlayer 的createMessage方法、配置消息并使用PlayerMessage.send将其发送到组件来传递配置更改。将消息发送到播放线程以确保它们与在播放器上执行的任何其他操作按顺序执行。