Flutter的启动和渲染流程

news2024/12/27 2:54:45

一、Flutter的启动流程

1、Flutter启动初始化

在安卓平台上启动App时,会先创建应用进程,然后创建FlutterApplication并执行其onCreate方法。在FlutterApplication的onCreate方法中,会执行Flutter相关的初始化,主要包括Flutter编译产物相关名称和路径的初始化、处理平台侧和Flutter侧Vsync信号的同步、加载引擎代码libflutter.so库、为Flutter创建存储目录等操作。

VsyncWaiter的初始化方法中,创建了原生平台层和Flutter引擎层同步Vsync信号的代理对象asyncWaitForVsyncDelegate。当Flutter侧需要刷新UI界面时,通过asyncWaitForVsyncDelegate和FlutterJNI将原生平台收到的Vsync回调转发给自身,然后在触发渲染流程更新界面。

下图是启动初始化的调用流程图:

在这里插入图片描述

2、FlutterActivity引擎获取

在FlutterActivity的onCreate方法中,通过调用delegate的onAttach方法获取Flutter引擎。引擎的获取主要分为三个步骤:

  1. 首先,从FlutterEngineCache中尝试获取已缓存的FlutterEngine
  2. 如果缓存中没有可用的FlutterEngine,则从provideFlutterEngine方法中获取自定义的FlutterEngine
  3. 如果以上两种方式都无法获取到FlutterEngine,那么将通过FlutterEngineGroup创建新的FlutterEngine

下图是FlutterActivity引擎获取的调用流程图:

在这里插入图片描述

3、FlutterEngineGroup创建引擎

Flutter引擎的创建有两种方式,一种是直接创建,一种是通过EngineGroup创建。通过EngineGroup方式创建的引擎之间能够共享部分资源(例如 GPU 上下文、字体度量和隔离线程的快照),从而加快首次渲染的速度、降低延迟并降低内存占用,每增加一个引擎只会增加约 180K 的内存占用。

FlutterEngineGroup创建引擎时,如果还没有创建过引擎,那么就会直接创建第一个引擎。如果已经有存在的引擎了,那么就会通过spawn部分资源共享的方式创建新引擎。

创建引擎时,会创建Dart侧的执行入口DartEntrypoint,如果提供自定义入口函数了,那DartEntrypoint中包含的就是自定义的入口函数名。如果没有提供自定义的入口函数,那么DartEntrypoint中包含的就是默认的入口函数名“main”。

创建引擎时,还会注册平台消息handler,创建平台系统channel,用于Flutter端和原生端通信。

创建引擎时,还会初始化Flutter环境、DartVM、FlutterEngine,关联FlutterJNI和Engine便于平台和引擎通信。创建FlutterRender对象,最终会执行Dart侧的入口方法,执行Dart代码。

下图是通过EngineGroup方式创建引擎的调用流程图:

在这里插入图片描述

二、Flutter的渲染流程

1、FlutterActivity视图渲染

FlutterActivity通过FlutterView呈现Flutter画面,并实现渲染显示。FlutterView继承自FrameLayout,它本身也是一个View。实际上,FlutterView的渲染是通过其内部的FlutterSurfaceView或FlutterTextureView(两者都是RenderSurface的实现类,不考虑FlutterImageView)来真正进行的。选择使用FlutterSurfaceView还是FlutterTextureView是根据RenderMode来决定的。

在FlutterView与Flutter引擎建立关联时,FlutterSurfaceView或FlutterTextureView也会与引擎的FlutterRenderer建立关联。Dart侧编写的Widget渲染树会交给引擎进行渲染,而FlutterRenderer负责将Flutter引擎的画面数据渲染到FlutterSurfaceView或FlutterTextureView上。渲染数据是通过FlutterJNI进行Engine与Android层之间传递的,最终将画面显示到原生平台视图中。

下图是FlutterActivity视图渲染的调用流程图:

在这里插入图片描述

2、Dart树渲染流程

Dart程序的执行是从入口方法(默认为main方法)开始的,接着会调用WidgetsBinding的scheduleAttachRootWidget方法,根据Dart业务代码构建出三棵树。然后,通过调用SchedulerBinding的scheduleWarmUpFrame方法启动渲染流程。最终,在RendererBinding的drawFrame方法中,将会调用PipelineOwner进行布局、合成、绘制等一系列操作。最后,通过调用Skia将视图绘制到屏幕上。

当Widget的状态需要变更时,我们可以调用State.setState来触发这个过程。首先,将会对发生变化的元素进行标记,并将其添加到_dirtyElements这个脏元素列表中。然后,通知Flutter引擎需要刷新用户界面。在Flutter引擎层,会向平台层注册Vsync信号回调。一旦Vsync信号到达,就会开始触发渲染流程。首先会对_dirtyElements中的元素进行重建和更新,生成更新后的渲染树。最终,在RendererBinding的drawFrame方法中,将会调用PipelineOwner进行布局、合成、绘制等一系列操作。最后,通过调用Skia将视图绘制到屏幕上。

下图是Dart树渲染的调用流程图:

在这里插入图片描述

【个人微信公众号】
专注于Android开发领域技术分享
在这里插入图片描述

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

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

相关文章

VB过程的递归调用,求4的阶乘

VB过程的递归调用,求4的阶乘 过程的递归调用,求4的阶乘 Private Function fact(n As Integer) As IntegerIf n 1 Thenfact 1Elsefact n * fact(n - 1)End If End Function Private Sub Command1_Click()Dim n As Integern 4Print fact(n) End Sub

【活动回顾】Rust:构建新时代基础设施的首选语言 @Qcon

以「启航 • AIGC 软件工程变革」为主题的 QCon 全球软件开发大会北京站于 9 月 5 日在北京富力万丽酒店圆满落幕!此次大会包含向量数据库、云原生、异构计算、面向 AI 的存储、微服务架构治理、FinOps 等近 30 个精彩专题。Databend Labs 作为深耕云原生数据库领域…

uniapp实现表格冻结

效果图如下: 思路: 1.由于APP项目需要,起初想去插件市场直接找现成的,结果找了很久没找到合适的(有的不支持vue2有的不能都支持APP和小程序) 2.后来,就只能去改uni-table源码了,因…

Vue3中watch用法

在 Vue3 中的组合式 API 中,watch 的作用和 Vue2 中的 watch 作用是一样的,他们都是用来监听响应式状态发生变化的,当响应式状态发生变化时,都会触发一个回调函数。 当需要在数据变化时执行异步或开销较大的操作时,com…

推三返一模式:电商运营的新趋势

商行业在近年来发展迅速,各种平台和产品层出不穷,竞争也越来越激烈。如何在这样的环境中脱颖而出,吸引和留住用户,提高销售和利润,是每一个电商企业家都面临的挑战。有没有一种运营模式,可以让用户主动购买…

解决报错:npm ERR! code 1

我是 npm install --legacy-peer-deps 成功了 解决方案: 升级swiper库:你可以尝试升级你的项目中的swiper库到5.2.0或更高版本,以满足vue-awesome-swiper的需求。你可以使用以下命令来进行升级: npm install swiperlatest 注意…

MPLS VPN跨域A

拓扑设计 拓扑介绍 如图,上海分公司与山东分公司之间为保证业务可以互通,需要使用MPLS VPN技术进行连接。中间R3与R4之间运行IGP,使用IGP传递路由,因为网络需要经过联通与移动两个AS域,所以使用MPLS VPN OptionA方案来…

SDXL prompt 笔记

模型 模型有两个,分别是 stable-diffusion-xl-base-1.0、stable-diffusion-xl-refiner-1.0。 base 模型是用来做文生图,refiner 模型是用来做图生图的。 SDXL 模型之 base、refiner 和 VAE_云水木石的博客-CSDN博客 分辨率 默认是1024*1024&#xf…

SAP MM学习笔记33 - 请求书照合中修改 带勘定设定Category(科目分配)的发票

请求书照合,对照的是 客户或供应商发过来发票。 带勘定设定Category(科目分配)的发票,一般是消耗品。 不过是否是消耗品,对供应商来说,跟他们没关系,供应商只是卖商品给你,怎么用是…

【建造者模式】

🏅我是默,一个在CSDN分享笔记的博主。📚📚 🌟在这里,我要推荐给大家我的专栏《20种Java设计模式》。🎯🎯 🚀无论你是编程小白,还是有一定基础的程序员&#x…

【AIGC】文心一言 - 生成中秋祝福语、诗词、图像、相关代码等

文心一言 - 生成中秋祝福词、诗、画 引言一、中秋祝福语二、中秋诗词三、中秋图像 - 闪亮的月亮四、中秋代码 引言 文心一言是百度推出国产聊天GPT产品,不仅可以与人对话互动,回答问题,协助创作,还能进行数据分析、逻辑判断、绘画…

代码随想录算法训练营day53|1143.最长公共子序列|1035.不相交的线|53. 最大子序和

1143.最长公共子序列 力扣题目链接 给定两个字符串 text1 和 text2,返回这两个字符串的最长公共子序列的长度。 一个字符串的 子序列 是指这样一个新的字符串:它是由原字符串在不改变字符的相对顺序的情况下删除某些字符(也可以不删除任何…

【链表】反转链表 II-力扣 92 题

💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:kuan 的首页,持续学…

【Linux】线程池 | 自旋锁 | 读写锁

文章目录 一、线程池1. 线程池模型和应用场景2. 单例模式实现线程池(懒汉模式) 二、其他常见的锁1. STL、智能指针和线程安全2. 其他常见的锁 三、读者写者问题1. 读者写者模型2. 读写锁 一、线程池 1. 线程池模型和应用场景 线程池是一种线程使用模式。线程过多会带来调度开…

Linux安装kafka-manager

相关链接https://github.com/yahoo/kafka-manager/releases kafka-manager-2.0.0.2下载地址 百度云链接:https://pan.baidu.com/s/1XinGcwpXU9YBF46qkrKS_A 提取码:tzvg 一、安装部署 1.把kafka-manager-2.0.0.2.zip拷贝到目录 /opt/app/elk 2.解压…

【Python】让Anaconda不再下载慢下载失败,Anaconda的下载源更换为国内源(保姆级图文)

目录 对于源的添加与修改1. 查看源3. 移除源4. 恢复默认源 检查添加源是否生效清理原来的索引配置总结 欢迎关注 『Python』 系列,持续更新中 欢迎关注 『Python』 系列,持续更新中 对于源的添加与修改 1. 查看源 conda config --show-sources## 2. 添…

天宇微纳芯片ic测试软件如何测试芯片上下电功能?

芯片的上电与下电功能测试是集成电路生产和研发过程中的关键环节,可以帮助企业确保产品的可靠性、整合性和兼容性,同时提高生产效率和产品质量。 因此在芯片的研发设计中,企业会对芯片的上下电有严格的要求,包括上下电的时序&…

飞腾主板显卡接eDP屏,显示花屏问题

故障现象: 开机过程中,在系统启动前无显示输出的阶段中。会有随机的白色竖条显示,位置不固定,且会抖动。 硬件环境: FT-1500A/16+显卡e8860; eDP屏京东方NV156FHM-N42; eDP屏数据(DP0+/-、DP1+/-、AUX)接口e8860直出; eDP屏背光接口控制BL_EN、BL_PWM由主板CPLD…

主要的经典模拟电路图 你都完懂了吗?

经典模拟电路图是电子学中非常重要的概念之一,在各种电子设备中,我们都可以看到这些电路图的身影。 一、差分放大电路,具有电路对称性的特点,此特点可以起到稳定工作点的作用,被广泛用干直接耦合电路和测量电路的输入级…