ExoPlayer架构详解与源码分析(4)——整体架构

news2025/1/16 2:39:20

系列文章目录

ExoPlayer架构详解与源码分析(1)——前言
ExoPlayer架构详解与源码分析(2)——Player
ExoPlayer架构详解与源码分析(3)——Timeline
ExoPlayer架构详解与源码分析(4)——整体架构


文章目录

  • 系列文章目录
  • 前言
  • Player的实现
    • BasePlayer
    • ExoPlayer
  • 线程模型
  • 总结


前言

根据前篇ExoPlayer架构详解与源码分析(2)——Player,想要直接实现Player接口需要非常复杂的代码逻辑,都写在一个类里肯定不现实,需要通过更多层次的扩展简化来实现,当然ExoPlayer就是这么做的,本篇来讲讲的如何通过BasePlayer来简化设计以及ExoPlayer如何将整个复杂的设计划分给一个个子系统来完成的。

Player的实现

先来看下整体架构
在这里插入图片描述
Player接口经过了一层BasePlayer简化,和ExoPlayer扩展。然后由ExoPlayerImpl实现,ExoPlayerImpl内部又依赖ExoPlayerImplInternal,ExoPlayerImplInternal再依据功能划分将任务交由各个组件,主要为MediaSource、Renderer、TrackSelector、LoadControl四大组件。

BasePlayer

先说BasePlayer 是个抽象类,主要作用是简化了Player接口的部分功能。

  • 实现了单文件列表增删改等操作,通过将单个MediaItem转为List,交由xxMediaItems实现。

      @Override
      public final void setMediaItem(MediaItem mediaItem) {
        setMediaItems(ImmutableList.of(mediaItem));
      }
    
  • 实例化出Timeline 中的 Window对象,这里主要用于Timeline getWindow 方法时装填的容器,因为Timeline 本身不持有Window或者Period,Timeline获取Window或者Period时都需要传入一个容器去获取,通过调用容器的set方法给容器赋值。

      protected BasePlayer() {
        window = new Timeline.Window();
      }
      
      @Override
      public final long getContentDuration() {//获取播放的总时长
        Timeline timeline = getCurrentTimeline();//先获取Timeline 由子类实现
        return timeline.isEmpty()
            ? C.TIME_UNSET//将初始化的window对象传入,方法里会将window对象赋值
            : timeline.getWindow(getCurrentMediaItemIndex(), window).getDurationMs();
      }
    
  • 实现了Player关于播放列表管理的设计,将MediaItem 播放列表查询相关交由Timeline管理,从这里可以看出上面针对MediaItem 的增删改,最终都是会封装到或者同步到Timeline里的,这里后面看到具体实现。

    
      @Override
      public final int getNextMediaItemIndex() {
        Timeline timeline = getCurrentTimeline();
        return timeline.isEmpty()
            ? C.INDEX_UNSET
            : timeline.getNextWindowIndex(
                getCurrentMediaItemIndex(), getRepeatModeForNavigation(), getShuffleModeEnabled());
      }
      
      @Override
      @Nullable
      public final MediaItem getCurrentMediaItem() {
        Timeline timeline = getCurrentTimeline();
        return timeline.isEmpty()
            ? null
            : timeline.getWindow(getCurrentMediaItemIndex(), window).mediaItem;
      }
    
      @Override
      public final int getMediaItemCount() {
        return getCurrentTimeline().getWindowCount();
      }
    
      @Override
      public final MediaItem getMediaItemAt(int index) {
        return getCurrentTimeline().getWindow(index, window).mediaItem;
      }
    
  • 基于Timeline将各种媒体的导航操作,如上一曲,下一曲,SEEK等,统一到自己抽象出的一个seekTo方法中。

    
      @Override
      public final void seekToNextMediaItem() {
        seekToNextMediaItemInternal(Player.COMMAND_SEEK_TO_NEXT_MEDIA_ITEM);
      }
      
      private void seekToNextMediaItemInternal(@Player.Command int seekCommand) {
        int nextMediaItemIndex = getNextMediaItemIndex();
        if (nextMediaItemIndex == C.INDEX_UNSET) {
          return;
        }
        if (nextMediaItemIndex == getCurrentMediaItemIndex()) {
          repeatCurrentMediaItem(seekCommand);
        } else {
          seekToDefaultPositionInternal(nextMediaItemIndex, seekCommand);
        }
      }
      
      private void repeatCurrentMediaItem(@Player.Command int seekCommand) {
        seekTo(
            getCurrentMediaItemIndex(),
            /* positionMs= */ C.TIME_UNSET,
            seekCommand,
            /* isRepeatingCurrentItem= */ true);
      }
      
      private void seekToDefaultPositionInternal(int mediaItemIndex, @Player.Command int seekCommand) {
        seekTo(
            mediaItemIndex,
            /* positionMs= */ C.TIME_UNSET,
            seekCommand,
            /* isRepeatingCurrentItem= */ false);
      }
    
       @Override
      public final int getNextMediaItemIndex() {
        Timeline timeline = getCurrentTimeline();
        return timeline.isEmpty()
            ? C.INDEX_UNSET//通过Timeline获取下一个索引
            : timeline.getNextWindowIndex(
                getCurrentMediaItemIndex(), getRepeatModeForNavigation(), getShuffleModeEnabled());
      }
      
      /**
       * Seek到指定的MediaItem中的指定位置
       *
       * @param mediaItemIndex MediaItem 的索引,可以理解成播放列表中的第几个
       * @param positionMs MediaItem 中的位置
       * @param seekCommand Seek 的类型用于权限控制,这里可以不用考虑
       * @param isRepeatingCurrentItem 是否重复当前播放项目
       */
      public abstract void seekTo(
          int mediaItemIndex,
          long positionMs,
          @Player.Command int seekCommand,
          boolean isRepeatingCurrentItem);
          
    
  • 完成了其他一些可以通过已有方法实现的方法。

      //判断当前命令是否可用,对应Player设计的第2点
      @Override
      public final boolean isCommandAvailable(@Command int command) {
        return getAvailableCommands().contains(command);//通过已有的getAvailableCommands来实现,getAvailableCommands由子类实现
      }
      //播放和暂停,实现了Player关于playWhenReady的设计,playWhenReady就是一个标记位,标记用户的一个播放意图
      //所以这里的play并不是立即开始播放的意思,而是调用者希望开始播放,实际播放要等到PlaybackState=STATE_READY的时候,pause同上
      @Override
      public final void play() {
        setPlayWhenReady(true);
      }
      //实现了Player关于isPlaying的设计
      @Override
      public final boolean isPlaying() {
        return getPlaybackState() == Player.STATE_READY
            && getPlayWhenReady()
            && getPlaybackSuppressionReason() == PLAYBACK_SUPPRESSION_REASON_NONE;
      }
    
      //获取直播流的延时
      @Override
      public final long getCurrentLiveOffset() {
        Timeline timeline = getCurrentTimeline();
        if (timeline.isEmpty()) {
          return C.TIME_UNSET;
        }
        long windowStartTimeMs =
            timeline.getWindow(getCurrentMediaItemIndex(), window).windowStartTimeMs;
        if (windowStartTimeMs == C.TIME_UNSET) {
          return C.TIME_UNSET;
        }
        //获取当前播放时间和实际实际的差值,使用当前时间(取服务端的实时时间如果可用)-(播放开始时间+已播放位置【含广告】)
        return window.getCurrentUnixTimeMs() - window.windowStartTimeMs - getContentPosition();
      }
      //获取缓冲百分比
      @Override
      public final int getBufferedPercentage() {
        long position = getBufferedPosition();
        long duration = getDuration();
        return position == C.TIME_UNSET || duration == C.TIME_UNSET
            ? 0
            : duration == 0 ? 100 : Util.constrainValue((int) ((position * 100) / duration), 0, 100);
      }
    

综上所述,BasePlay实现了部分Player接口的设计,简化了Player接口实现,为后续的子类铺平道路。

ExoPlayer

一个接口定义,继承扩展了Player接口,实现MediaSource的播放。ExoPlayer播放器本体设计都在这里了,将Player接口复杂的设计,通过建立一个运行框架,将功能分散到各个子系统,协调这些子系统完成播放器的播放等最初的设计目标。
我们将文章开头的架构图进一步扩充下。
在这里插入图片描述

ExoPlayer 设计理念之一就是高度可定制化,主要任务是协调各个组件间工作,而对媒体的类型、存储方式、加载方式、如何展示等并不关心。ExoPlayer并不直接实现媒体的加载与渲染,而是将这些工作交给播放器创建或者准备时注入的组件,这些组件包括:

  • MediaSource
    • 主要作用是定义需要播放的媒体基本信息、加载媒体以及定义了从哪里读取已经加载的媒体数据。
    • 通过将MediaItems传入MediaSource.Factory(播放器创建时指定)创建,也可以直接调用setMediaSource方法创建。
    • 播放器默认提供了 DefaultMediaSourceFactory可以根据不同类型的MediaItem创建出不同的MediaSource,包括progressive ,HLS,DASH,SmoothStreaming 。
  • Renderers
    • 包含了用于渲染媒体的各个组件。
    • 提供了像MediaCodecVideoRenderer, MediaCodecAudioRenderer, TextRenderer and MetadataRenderer这些组件用于常见媒体的渲染。
    • Renderer 使用MediaSource提供的数据来渲染。
    • 可以通过ExoPlayer接口提供的getRendererCount获取渲染器的数量,getRendererType获取各自轨道类型。
  • TrackSelector
    • 用于选择由MediaSource提供的可用于渲染器的轨道。
    • 播放器在创建时默认注入了DefaultTrackSelector,可以用于大部分情况的轨道选择 。
  • LoadControl
    • 主要用于控制MediaSource何时缓冲更多媒体数据以及缓冲多少数据。
    • 播放器在创建时默认注入了DefaultLoadControl,可以用于大部分情况的数据加载 。

上面的组件在创建ExoPlayer 时都会注入一个默认的实现,当默认组件无法满足需求时,可以通过自定义的组件来构建播放器。如可以通过设置自定义的LoadControl来更改播放器默认的缓存加载策略,或者通过添加子当以的Renderer来支持Android本身不支持的视频编码格式。

上图可以看到不光ExoPlayer使用了注入组件的概念,上面列出ExoPlayer组件本身就和ExoPlayer一样也使用了组件注入的概念,这些组件本身也是由子组件注入创建而来的,将这些的组件本地的功能又再一次细化分配给各自的子组件来完成,并且这些子组件同样也支持自定义。如上图,默认的在创建MediaSource时就需要注入一个或者多个DataSource 工厂,通过提供不同的DataSource工厂,可以从不同的数据源加载数据。基于这种设计思路下的系统共同打造了一个高度可定制化的ExoPlayer。

线程模型

下图展示了ExoPlayer的线程模型
在这里插入图片描述
可以看出播放器线程主要分为3部分

  • application thread
    • 应用线程只有一个 ,大部分情况是应用的主线程,对应Android的UI线程。
    • 如果使用了ExoPlayer 的UI库或者IMA库也要使用应用的主线程。
    • 可以通过在创建播放器时传递“Looper”来显式指定用于访问 ExoPlayer 实例的线程,如果未指定“Looper”,则使用创建播放器的线程的“Looper”,或者如果该线程没有“Looper”,则使用应用程序主线程的“Looper”。无论哪种情况,都要可以通过Player接口定义的getApplicationLooper获取到访问播放器线程的“Looper”。
    • 由于是主线程应用可以直接在主线程中获取播放器的相关信息,这些信息通常保存在ExoPlayerImpl中无需异步回调即可立刻获取到数据,这也符合ExoPlayer架构详解与源码分析(2)——Player中关于Player的设计。
    • 已注册的监听都是在主线程(通过getApplicationLooper获取)中回调的,这就意味着组测这些监听的地方也必须在同一个主线程中。对于监听类的回调这些都是异步的,这个回调最终会使用主线程的Handler分发到主线程里,这也是为什么创建ExoPlayer是必须要指定主线程的原因。
  • internal playback thread
    • 一个播放器实例只有一个,主要负责播放。renderer、MediaSources、TrackSelectors 和 LoadControls 等注入到播放器组件都是在这个线程里调用的。
    • 这个线程也是一个Looper线程,有一个Handler用于将主线程的请求发送到Looper里进行分发。
    • 当应用程序在播放器上执行操作(如Seek)时,消息会通过主线程持有的Handler发送到内部播放线程的Looper然后分发到内部线程里,并在内部播放线程里调用相关方法执行相应的操作。类似地,当内部播放线程上发生播放事件时,消息将通过另一个Handler分发到主线程。主线程使用队列中的消息,更新应用程序可见状态并调用相应的监听回调。
    • 这部分Exoplayer实现在ExoPlayerImplInternal中,在其初始化过程中创建了一个HandlerThread来实现后面会讲到。
  • background threads
    • 各个注入到ExoPlayer中组件的后台线程,会有多个。
    • 注入的播放器组件可以使用额外的后台线程执行任务。例如,MediaSource 可以使用后台线程来加载数据。这些线程都是由不同的MediaSource实现决定的。

总结

可以看到EoxPlayer架构的高度可定制化,基本每一个组件都可以在创建时自定义,然后注入到播放器中实现自定义的播放器。
EoxPlayer这些设计在后续的分析中都会体现,按顺序下篇应该了解下ExoPlayerImpl和ExoPlayerImplInternal,但是他们中很多功能都是依赖于4大组件的,而且4大组件直接又是相互独立的,所以计划后面几篇先把它的4大组件分析下,最后通过分析ExoPlayerImpl和ExoPlayerImplInternal将前面将的4大组件串联起来,了解ExoPlayerImpl和ExoPlayerImplInternal是如何协调这些组件完成播放的。下篇预计先从最复杂的组件MediaSource开始分析。


版权声明 ©
本文为CSDN作者山雨楼原创文章
转载请注明出处
原创不易,觉得有用的话,收藏转发点赞支持

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1074217.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

0-1背包理论基础详解

0-1背包问题:有 n 种物品,每种物品只有1个,求解将哪些物品装入背包里物品价值总和最大。 图片来自:代码随想录 0-1背包问题举例: 题目描述: 使用二维dp数组解决背包问题 动规五部曲: dp[i][j…

智能工厂MES系统,终端设备支持手机、PDA、工业平板、PC

一、开源项目简介 源计划智能工厂MES系统(开源版) 功能包括销售管理,仓库管理,生产管理,质量管理,设备管理,条码追溯,财务管理,系统集成,移动端APP。 二、开源协议 使用GPL-3.0开…

【python海洋专题十四】读取多个盐度nc数据画盐度季节变化图

本期内容 读取多个盐度文件;拼接数据在画盐度的季节分布图Part01. 使用数据 IAP 网格盐度数据集 数据详细介绍: 见文件附件: pages/file/dl?fid378649712527544320 全球温盐格点数据.pdf IAP_Global_ocean_gridded_product.pdf 全球温…

Android多线程学习:线程池(一)

一、概念 线程池:创建并维护一定数量的空闲线程,当有需要执行的任务,就交付给线程池中的一个线程,任务执行结束后,该线程也不会死亡,而是回到线程池中重新变为空闲状态。 线程池优点: 1、重用…

Linux系列---【查看mac地址】

查看mac地址命令 查看所有网卡命令 nmcli connection show 查看物理网卡mac地址 ifconfig 删除网卡 nmcli connection delete virbr0 禁用libvirtd.service systemctl disable libvirtd.service 启用libvirtd.service systemctl enable libvirtd.service

使用css 与 js 两种方式实现导航栏吸顶效果

position的属性我们一般认为有 position:absolute postion: relative position:static position:fixed position:inherit; position:initial; position:unset; 但是我最近发现了一个定位position:sticky 这个可以称为粘性定位。 这个粘性定位的元素会始终在那个位置 <st…

firefox的主题文件位置在哪?记录以防遗忘

这篇文章写点轻松的 最近找到了一个自己喜欢的firefox主题,很想把主题的背景图片找到,所以找了下主题文件所在位置 我的firefox版本:版本: 118.0.1 (64 位)主题名称: Sora Kawai 我的位置在 C:\Users\mizuhokaga\AppData\Roaming\Mozilla\Firefox\Profiles\w0e4e24v.default…

可视大盘 + 健康分机制,火山引擎 DataLeap 为企业降低资源优化门槛!

更多技术交流、求职机会&#xff0c;欢迎关注字节跳动数据平台微信公众号&#xff0c;回复【1】进入官方交流群 随着数仓及研发技术团队维护的数据量大、资源使用量大、成本越高、优化压力越大。如何主动发现无效或低效使用的资源&#xff0c;并且可以周期性高效的进行主动治理…

js 气泡上升和鼠标点击事件

效果图 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta name"viewport" content"widthdevice-width, initial-scale1.0" /><title>Document</title><style>bod…

银河麒麟服务器x86安装ntp客户端,并配置成功可以同步时间

脚本 # 安装ntp客户端 sudo dnf install chrony # 配置 pool 2.centos.pool.ntp.org iburst给这一行加注释 sudo sed -i s/^pool 2.centos.pool.ntp.org iburst/#&/ /etc/chrony.conf # 添加3个阿里云NTP服务器 # echo -e "server ntp1.aliyun.com iburst\nserver nt…

安卓 kotlin-supportFragmentManager报红

如果你继承baseActivity 请查看 是不是继承 AppCompatActivity

全网都在找的python+requests接口自动化测试框架实例详解教程

前言 Python是一种功能强大的编程语言&#xff0c;它可以用于自动化测试&#xff0c;特别是接口自动化测试。许多Python库都可以用于接口自动化测试&#xff0c;其中requests库是其中最受欢迎的库之一。 requests库可以用于发送HTTP请求并获取服务器响应&#xff0c;从而轻松…

java案例21:学生管理系统

思路&#xff1a; 编写一个学生管理系统&#xff0c; 实现对学生信息的添加、删除、修改和查询功能。首页&#xff1a; 用于显示系统的所有操作&#xff0c;并根据用户在控制台的输入选择需要使用的功能查询功能&#xff1a; 用户选择后&#xff0c;在控制台打印所有学生信息添…

阿里云服务器公网带宽多少钱1M?

阿里云服务器公网带宽计费模式按固定带宽”计费多少钱1M&#xff1f;地域不同带宽价格不同&#xff0c;北京、杭州、深圳等大陆地域价格是23元/Mbps每月&#xff0c;中国香港1M带宽价格是30元一个月&#xff0c;美国硅谷是30元一个月&#xff0c;日本东京1M带宽是25元一个月&am…

爬虫Python

文章目录 基本数据类型bytes类型python数据类型转换 python运算符&#xff08;必会&#xff01;&#xff01;&#xff01;&#xff09;python数字数学函数&#xff08;必会&#xff01;&#xff01;&#xff01;&#xff09;随机数函数三角函数&#xff08;简&#xff09;数字常…

c++视觉处理---均值滤波

均值滤波 cv::blur()函数是OpenCV中用于应用均值滤波的函数。均值滤波是一种简单的平滑技术&#xff0c;它计算每个像素周围像素的平均值&#xff0c;并用该平均值替代原始像素值。这有助于降低图像中的噪声&#xff0c;并可以模糊图像的细节。 以下是cv::blur()函数的基本用…

记一次 .NET某账本软件 非托管泄露分析

一&#xff1a;背景 1. 讲故事 中秋国庆长假结束&#xff0c;哈哈&#xff0c;在老家拍了很多的短视频&#xff0c;有兴趣的可以上B站观看&#xff1a;https://space.bilibili.com/409524162 &#xff0c;今天继续给大家分享各种奇奇怪怪的.NET生产事故&#xff0c;希望能帮助…

c++视觉处理---高斯滤波

高斯滤波处理 高斯滤波是一种常用的平滑滤波方法&#xff0c;它使用高斯函数的权重来平滑图像。高斯滤波通常用于去除噪声并保留图像中的细节。在OpenCV中&#xff0c;可以使用cv::GaussianBlur()函数来应用高斯滤波。 以下是cv::GaussianBlur()函数的基本用法&#xff1a; …

区块链技术的飞跃: 2023年的数字革命

随着时代的推进和技术的不断创新&#xff0c;2023年成为区块链技术飞跃发展的一年。区块链&#xff0c;一个曾经只是数字货币领域的技术&#xff0c;现在已经逐渐渗透到各个行业&#xff0c;成为推动数字经济发展的重要力量。在这个数字革命的时代&#xff0c;我们探讨区块链技…

纸黄金效率太低不如做现货

如果从字面上的意义去理解&#xff0c;纸黄金就是在账面上交易的黄金&#xff0c;具体来说它是国内银行为客户提供的一种记账式的黄金买卖&#xff0c;交易的记录和买卖的盈亏值&#xff0c;都只会在预先开设的账户上体现&#xff0c;投资过程中不涉及实物黄金的交收。 对于追求…