HLS

ExoPlayer는 여러 컨테이너 형식으로 HLS를 지원합니다. 포함된 오디오 및 동영상 샘플 형식도 지원되어야 합니다 (자세한 내용은 샘플 형식 섹션 참고). HLS 콘텐츠 제작자는 이 블로그 게시물에 설명된 대로 고품질 HLS 스트림을 생성하는 것이 좋습니다.

기능 지원됨 비고
컨테이너
MPEG-TS
FMP4/CMAF
ADTS (AAC)
MP3
자막 / 자막
CEA-608
CEA-708
WebVTT
메타데이터
ID3
SCTE-35 아니요
콘텐츠 보호
AES-128
샘플 AES-128 아니요
Widevine API 19 이상 ('cenc' 스키마) 및 25 이상 ('cbcs' 스키마)
PlayReady SL2000 Android TV만
서버 제어
델타 업데이트
재생목록 새로고침 차단
미리 로드 힌트 로드 차단 길이가 정의되지 않은 바이트 범위는 예외입니다.
광고 삽입
서버 가이드 광고 삽입(전면 광고) 일부만 X-ASSET-URI가 있는 VOD만 해당 라이브 스트림 및 X-ASSET-LIST는 나중에 추가됩니다.
IMA 서버 측 및 클라이언트 측 광고 광고 삽입 가이드
실시간 재생
일반 라이브 재생
지연 시간이 짧은 HLS (Apple)
지연 시간이 짧은 HLS (커뮤니티) 아니요
일반 미디어 클라이언트 데이터 CMCD CMCD 통합 가이드

MediaItem 사용

HLS 스트림을 재생하려면 HLS 모듈을 사용해야 합니다.

Kotlin

implementation("androidx.media3:media3-exoplayer-hls:1.6.0")

Groovy

implementation "androidx.media3:media3-exoplayer-hls:1.6.0"

그런 다음 HLS 재생목록 URI의 MediaItem를 만들고 이를 플레이어에 전달할 수 있습니다.

Kotlin

// Create a player instance.
val player = ExoPlayer.Builder(context).build()
// Set the media item to be played.
player.setMediaItem(MediaItem.fromUri(hlsUri))
// Prepare the player.
player.prepare()

자바

// Create a player instance.
ExoPlayer player = new ExoPlayer.Builder(context).build();
// Set the media item to be played.
player.setMediaItem(MediaItem.fromUri(hlsUri));
// Prepare the player.
player.prepare();

URI가 .m3u8로 끝나지 않으면 MediaItem.BuildersetMimeTypeMimeTypes.APPLICATION_M3U8를 전달하여 콘텐츠 유형을 명시적으로 표시할 수 있습니다.

미디어 항목의 URI는 미디어 재생목록 또는 대안 재생목록을 가리킬 수 있습니다. URI가 여러 #EXT-X-STREAM-INF 태그를 선언하는 다중 변형 재생목록을 가리키는 경우 ExoPlayer는 사용 가능한 대역폭과 기기 기능을 모두 고려하여 변형 간에 자동으로 조정합니다.

HlsMediaSource 사용

더 많은 맞춤설정 옵션을 사용하려면 MediaItem 대신 HlsMediaSource를 만들어 플레이어에 직접 전달하면 됩니다.

Kotlin

// Create a data source factory.
val dataSourceFactory: DataSource.Factory = DefaultHttpDataSource.Factory()
// Create a HLS media source pointing to a playlist uri.
val hlsMediaSource =
  HlsMediaSource.Factory(dataSourceFactory).createMediaSource(MediaItem.fromUri(hlsUri))
// Create a player instance.
val player = ExoPlayer.Builder(context).build()
// Set the HLS media source as the playlist with a single media item.
player.setMediaSource(hlsMediaSource)
// Prepare the player.
player.prepare()

자바

// Create a data source factory.
DataSource.Factory dataSourceFactory = new DefaultHttpDataSource.Factory();
// Create a HLS media source pointing to a playlist uri.
HlsMediaSource hlsMediaSource =
    new HlsMediaSource.Factory(dataSourceFactory).createMediaSource(MediaItem.fromUri(hlsUri));
// Create a player instance.
ExoPlayer player = new ExoPlayer.Builder(context).build();
// Set the HLS media source as the playlist with a single media item.
player.setMediaSource(hlsMediaSource);
// Prepare the player.
player.prepare();

매니페스트 액세스

Player.getCurrentManifest를 호출하여 현재 매니페스트를 가져올 수 있습니다. HLS의 경우 반환된 객체를 HlsManifest로 변환해야 합니다. Player.ListeneronTimelineChanged 콜백은 매니페스트가 로드될 때마다 호출됩니다. 주문형 콘텐츠의 경우 한 번, 라이브 콘텐츠의 경우 여러 번 발생할 수 있습니다. 다음 코드 스니펫은 매니페스트가 로드될 때마다 앱이 어떤 작업을 실행할 수 있는지 보여줍니다.

Kotlin

player.addListener(
  object : Player.Listener {
    override fun onTimelineChanged(timeline: Timeline, @TimelineChangeReason reason: Int) {
      val manifest = player.currentManifest
      if (manifest is HlsManifest) {
        // Do something with the manifest.
      }
    }
  }
)

자바

player.addListener(
    new Player.Listener() {
      @Override
      public void onTimelineChanged(
          Timeline timeline, @Player.TimelineChangeReason int reason) {
        Object manifest = player.getCurrentManifest();
        if (manifest != null) {
          HlsManifest hlsManifest = (HlsManifest) manifest;
          // Do something with the manifest.
        }
      }
    });

전면 광고로 HLS 스트림 재생

HLS 사양은 미디어 재생목록에 전면 광고 정보를 포함하는 데 사용할 수 있는 HLS 전면 광고를 정의합니다. ExoPlayer는 기본적으로 이러한 전면 광고를 무시합니다. HlsInterstitialsAdsLoader를 사용하여 지원을 추가할 수 있습니다. 처음부터 사양의 모든 기능이 지원되는 것은 아닙니다. 스트림에 대한 지원이 누락된 경우 GitHub에서 문제를 제출하여 알려주시고 스트림 URI를 보내주시면 스트림 지원을 추가해 드리겠습니다.

playlist API와 함께 MediaItem 사용

전면 광고와 함께 HLS 스트림을 재생하는 가장 편리한 방법은 HlsInterstitialsAdsLoader.AdsMediaSourceFactory로 ExoPlayer 인스턴스를 빌드하는 것입니다. 이렇게 하면 Player 인터페이스의 MediaItem 기반 재생목록 API를 사용하여 HLS 전면 광고를 재생할 수 있습니다.

ExoPlayerMediaSource.Factory는 플레이어 인스턴스를 빌드할 때 빌더에 삽입할 수 있습니다.

Kotlin

hlsInterstitialsAdsLoader = HlsInterstitialsAdsLoader(context)
// Create a MediaSource.Factory for HLS streams with interstitials.
var hlsMediaSourceFactory =
  HlsInterstitialsAdsLoader.AdsMediaSourceFactory(
    hlsInterstitialsAdsLoader,
    playerView,
    DefaultMediaSourceFactory(context),
  )

// Build player with interstitials media source factory
player =
  ExoPlayer.Builder(context)
    .setMediaSourceFactory(hlsMediaSourceFactory)
    .build()

// Set the player on the ads loader.
hlsInterstitialsAdsLoader.setPlayer(player)
playerView.setPlayer(player)

자바

hlsInterstitialsAdsLoader = new HlsInterstitialsAdsLoader(context);
// Create a MediaSource.Factory for HLS streams with interstitials.
MediaSource.Factory hlsMediaSourceFactory =
      new HlsInterstitialsAdsLoader.AdsMediaSourceFactory(
          hlsInterstitialsAdsLoader, playerView, new DefaultMediaSourceFactory(context));

// Build player with interstitials media source factory
player =
    new ExoPlayer.Builder(context)
        .setMediaSourceFactory(hlsMediaSourceFactory)
        .build();

// Set the player on the ads loader.
hlsInterstitialsAdsLoader.setPlayer(player);
playerView.setPlayer(player);

이러한 플레이어 설정을 사용하면 HLS 전면 광고를 재생하는 것은 플레이어에 AdsConfiguration가 있는 미디어 항목을 설정하는 것과 같습니다.

Kotlin

player.setMediaItem(
  MediaItem.Builder()
    .setUri("https://www.example.com/media.m3u8")
    .setAdsConfiguration(
      AdsConfiguration.Builder(Uri.parse("hls://interstitials"))
        .setAdsId("ad-tag-0") // must be unique within playlist
        .build())
    .build())

player.prepare();
player.play();

자바

player.setMediaItem(
    new MediaItem.Builder()
        .setUri("https://www.example.com/media.m3u8")
        .setAdsConfiguration(
            new AdsConfiguration.Builder(Uri.parse("hls://interstitials"))
                .setAdsId("ad-tag-0") // must be unique within playlist
                .build())
        .build());
player.prepare();
player.play();

미디어 소스 기반 API 사용

또는 기본 미디어 소스 팩토리를 재정의하지 않고도 ExoPlayer 인스턴스를 빌드할 수 있습니다. 그러면 앱은 전면 광고를 지원하기 위해 HlsInterstitialsAdsLoader.AdsMediaSourceFactory를 직접 사용하여 MediaSource를 만들고 미디어 소스 기반 재생목록 API를 사용하여 ExoPlayer에 제공할 수 있습니다.

Kotlin

hlsInterstitialsAdsLoader = HlsInterstitialsAdsLoader(context)
// Create a MediaSource.Factory for HLS streams with interstitials.
var hlsMediaSourceFactory =
  HlsInterstitialsAdsLoader.AdsMediaSourceFactory(hlsInterstitialsAdsLoader, playerView, context)

// Build player with default media source factory.
player = new ExoPlayer.Builder(context).build();

// Create an media source from an HLS media item with ads configuration.
val mediaSource =
  hlsMediaSourceFactory.createMediaSource(
    MediaItem.Builder()
      .setUri("https://www.example.com/media.m3u8")
      .setAdsConfiguration(
        MediaItem.AdsConfiguration.Builder(Uri.parse("hls://interstitials"))
          .setAdsId("ad-tag-0")
          .build()
      )
      .build()
  )

// Set the media source on the player.
player.setMediaSource(mediaSource)
player.prepare()
player.play()

자바

HlsInterstitialsAdsLoader hlsInterstitialsAdsLoader = new HlsInterstitialsAdsLoader(context);
// Create a MediaSource.Factory for HLS streams with interstitials.
MediaSource.Factory hlsMediaSourceFactory =
    new HlsInterstitialsAdsLoader.AdsMediaSourceFactory(
      hlsInterstitialsAdsLoader, playerView, context);

// Build player with default media source factory.
player = new ExoPlayer.Builder(context).build();

// Create an media source from an HLS media item with ads configuration.
MediaSource mediaSource =
    hlsMediaSourceFactory.createMediaSource(
      new MediaItem.Builder()
        .setUri("https://www.example.com/media.m3u8")
        .setAdsConfiguration(
            new MediaItem.AdsConfiguration.Builder(Uri.parse("hls://interstitials"))
                .setAdsId("ad-tag-0")
                .build())
        .build());

// Set the media source on the player.
exoPlayer.setMediaSource(mediaSource);
exoPlayer.prepare();
exoPlayer.play();

광고 이벤트 수신

ListenerHlsInterstitialsAdsLoader에 추가하여 HLS 전면 광고 재생과 관련된 상태 변경에 관한 이벤트를 모니터링할 수 있습니다. 이를 통해 앱 또는 SDK는 재생된 광고, 로드 중인 확장 소재 목록, 준비 중인 광고 미디어 소스를 추적하거나 확장 소재 목록 로드 및 광고 준비 오류를 감지할 수 있습니다. 또한 광고 미디어 소스에서 내보낸 메타데이터를 세분화된 광고 재생 확인이나 광고 재생 진행 상황 추적을 위해 수신할 수 있습니다.

Kotlin

class AdsLoaderListener : HlsInterstitialsAdsLoader.Listener {

  override fun onStart(mediaItem: MediaItem, adsId: Any, adViewProvider: AdViewProvider) {
    // Do something when HLS media item with interstitials is started.
  }

  override fun onMetadata(
    mediaItem: MediaItem,
    adsId: Any,
    adGroupIndex: Int,
    adIndexInAdGroup: Int,
    metadata: Metadata,
  ) {
    // Do something with metadata that is emitted by the ad media source of the given ad.
  }

  override fun onAdCompleted(
    mediaItem: MediaItem,
    adsId: Any,
    adGroupIndex: Int,
    adIndexInAdGroup: Int,
  ) {
    // Do something when ad completed playback.
  }

  // ... See JavaDoc for further callbacks of HlsInterstitialsAdsLoader.Listener.

  override fun onStop(mediaItem: MediaItem, adsId: Any, adPlaybackState: AdPlaybackState) {
    // Do something with the resulting ad playback state when stopped.
  }
}

자바

private class AdsLoaderListener
    implements HlsInterstitialsAdsLoader.Listener {

  // implement HlsInterstitialsAdsLoader.Listener

  @Override
  public void onStart(MediaItem mediaItem, Object adsId, AdViewProvider adViewProvider) {
    // Do something when HLS media item with interstitials is started.
  }

  @Override
  public void onMetadata(
      MediaItem mediaItem,
      Object adsId,
      int adGroupIndex,
      int adIndexInAdGroup,
      Metadata metadata) {
    // Do something with metadata that is emitted by the ad media source of the given ad.
  }

  @Override
  public void onAdCompleted(
      MediaItem mediaItem, Object adsId, int adGroupIndex, int adIndexInAdGroup) {
    // Do something when ad completed playback.
  }

  // ... See JavaDoc for further callbacks

  @Override
  public void onStop(MediaItem mediaItem, Object adsId, AdPlaybackState adPlaybackState) {
    // Do something with the resulting ad playback state when stopped.
  }
}

사용 가능한 모든 콜백에 관한 자세한 문서는 HlsInterstitialsAdsLoader.Listener의 JavaDoc을 참고하세요.

그런 다음 리스너를 광고 로더에 추가할 수 있습니다.

Kotlin

var listener  = AdsLoaderListener();
// Add the listener to the ads loader to receive ad loader events.
hlsInterstitialsAdsLoader.addListener(listener);

자바

AdsLoaderListener listener = new AdsLoaderListener();
// Add the listener to the ads loader to receive ad loader events.
hlsInterstitialsAdsLoader.addListener(listener);

HlsInterstitialsAdsLoader 수명 주기

HlsInterstitialsAdsLoader 또는 HlsInterstitialsAdsLoader.AdsMediaSourceFactory 인스턴스는 광고를 로드해야 하는 여러 미디어 소스를 만드는 여러 플레이어 인스턴스에 재사용할 수 있습니다.

예를 들어 ActivityonCreate 메서드에서 인스턴스를 만든 다음 여러 플레이어 인스턴스에 재사용할 수 있습니다. 이는 단일 플레이어 인스턴스에서 동시에 사용하고 있는 경우에만 작동합니다. 이는 앱이 백그라운드로 전환되고 플레이어 인스턴스가 소멸된 후 앱이 다시 포그라운드로 전환될 때 새 인스턴스가 생성되는 일반적인 사용 사례에 유용합니다.

Kotlin

// Create the ads loader instance (for example onCreate).
hlsInterstitialsAdsLoader = HlsInterstitialsAdsLoader(context);

// Build a player and set it on the ads loader (for example onStart).
player = ExoPlayer.Builder(context).build();
hlsInterstitialsAdsLoader.setPlayer(player);

// Release the player and unset it on the ads loader (for example onStop).
player.release();
hlsInterstitialsAdsLoader.setPlayer(null);

// Build another player and set it on the ads loader (for example onStart).
player = ExoPlayer.Builder(context).build();
hlsInterstitialsAdsLoader.setPlayer(player);

// Release the player and unset it on the ads loader (for example onStop).
player.release();
hlsInterstitialsAdsLoader.setPlayer(null);

// Release the ads loader when not used anymore  (for example onDestroy).
hlsInterstitialsAdsLoader.release();

자바

// Create the ads loader instance (for example onCreate).
hlsInterstitialsAdsLoader = new HlsInterstitialsAdsLoader(context);

// Build a player and set it on the ads loader (for example onStart).
player = new ExoPlayer.Builder(context).build();
hlsInterstitialsAdsLoader.setPlayer(player);

// Release the player and unset it on the ads loader (for example onStop).
player.release();
hlsInterstitialsAdsLoader.setPlayer(null);

// Build another player and set it on the ads loader (for example onStart).
player = new ExoPlayer.Builder(context).build();
hlsInterstitialsAdsLoader.setPlayer(player);

// Release the player and unset it on the ads loader (for example onStop).
player.release();
hlsInterstitialsAdsLoader.setPlayer(null);

// Release the ads loader when not used anymore  (for example onDestroy).
hlsInterstitialsAdsLoader.release();

일반적으로 광고 로더에서 다음 플레이어 인스턴스를 설정하기 전에 이전 플레이어 인스턴스를 해제해야 합니다. 광고 로더 자체가 해제되면 광고 로더를 더 이상 사용할 수 없습니다.

재생 맞춤설정

ExoPlayer는 앱의 요구사항에 맞게 재생 환경을 조정할 수 있는 여러 가지 방법을 제공합니다. 예시는 맞춤설정 페이지를 참고하세요.

청크 없는 준비 사용 중지

기본적으로 ExoPlayer는 청크 없는 준비를 사용합니다. 즉, ExoPlayer는 대안 재생목록의 정보만 사용하여 스트림을 준비합니다. 이는 #EXT-X-STREAM-INF 태그에 CODECS 속성이 포함된 경우에 작동합니다.

미디어 세그먼트에 #EXT-X-MEDIA:TYPE=CLOSED-CAPTIONS 태그로 다중 변형 재생목록에 선언되지 않은 멀티플렉스 자막 트랙이 포함된 경우 이 기능을 사용 중지해야 할 수 있습니다. 그렇지 않으면 자막 트랙이 감지되고 재생되지 않습니다. 다음 스니펫과 같이 HlsMediaSource.Factory에서 청크 없는 준비를 사용 중지할 수 있습니다. 이렇게 하면 ExoPlayer가 이러한 추가 트랙을 검색하기 위해 미디어 세그먼트를 다운로드해야 하므로 시작 시간이 늘어나므로 대신 대안 재생목록에서 자막 트랙을 선언하는 것이 좋습니다.

Kotlin

val hlsMediaSource =
  HlsMediaSource.Factory(dataSourceFactory)
    .setAllowChunklessPreparation(false)
    .createMediaSource(MediaItem.fromUri(hlsUri))

자바

HlsMediaSource hlsMediaSource =
    new HlsMediaSource.Factory(dataSourceFactory)
        .setAllowChunklessPreparation(false)
        .createMediaSource(MediaItem.fromUri(hlsUri));

고품질 HLS 콘텐츠 제작

ExoPlayer를 최대한 활용하려면 HLS 콘텐츠를 개선하기 위해 따라야 할 몇 가지 가이드라인이 있습니다. 자세한 내용은 ExoPlayer의 HLS 재생에 관한 Medium 게시물을 참고하세요. 주요 사항은 다음과 같습니다.

  • 정확한 세그먼트 길이를 사용합니다.
  • 연속 미디어 스트림을 사용합니다. 세그먼트 간에 미디어 구조가 변경되지 않도록 합니다.
  • #EXT-X-INDEPENDENT-SEGMENTS 태그를 사용합니다.
  • 동영상과 오디오가 모두 포함된 파일보다는 디뮤싱된 스트림을 사용하는 것이 좋습니다.
  • 다변형 재생목록에 가능한 한 많은 정보를 포함합니다.

다음 가이드라인은 라이브 스트림에만 적용됩니다.

  • #EXT-X-PROGRAM-DATE-TIME 태그를 사용합니다.
  • #EXT-X-DISCONTINUITY-SEQUENCE 태그를 사용합니다.
  • 긴 실시간 기간을 제공합니다. 1분 이상이면 좋습니다.