从前面的文章"Flutter入口runApp源码分析"我们知道,在WidgrtsFlutterBinding初始化以后,接着就会调用到WidgetsBinding
的scheduleAttachRootWidget
方法,scheduleAttachRootWidget
方法内部直接异步调用了attachRootWidget
,这个方法就是三棵树的起源。
直接上源码太枯燥,我们先来个时序图,跟着主步骤一步步来。
==================================================================
根据上面的时序图我们能清楚的看到,关于Flutter widget三棵树的起源就是在WidgetsBind的attachRootWidget方法之中。
=================================================================================
void attachRootWidget(Widget rootWidget) {
_readyToProduceFrames = true;
// 创建RenderObjectToWidgetAdapter,RenderObjectToWidgetAdapter是RenderObjectElement和Element的桥梁。
// 第一步
_renderViewElement = RenderObjectToWidgetAdapter(
container: renderView,
debugShortDescription: ‘[root]’,
child: rootWidget,
//第二部
).attachToRenderTree(buildOwner!, renderViewElement as RenderObjectToWidgetElement?);
}
attachRootWidget方法很简单,主要分为两步:
-
创建RenderObjectToWidgetAdapter
-
调用RenderObjectToWidgetAdapter的attachToRenderTree方法
很明显核心逻辑都在RenderObjectToWidgetAdapter
,RenderObjectToWidgetAdapter
,其实也是一个widget,初始化的时候将rootWidget传递过去,作为child,还有个参数就是renderView,实际上是一个RenderObject。而RenderObjectToWidgetAdapter
作为一个adapter的主要作用就是建立RenderObject,Element,Widget之间的桥梁。
RenderView
就是渲染树的根。
刚刚讲到这里RenderObjectToWidgetAdapter
就已经持有了rootWidget和rootRender两棵树的根节点,下面我们再来看,attachToRenderTree方法。
===================================================================================
RenderObjectToWidgetElement attachToRenderTree(BuildOwner owner, [ RenderObjectToWidgetElement? element ]) {
if (element == null) {
owner.lockState(() {
//创建rootElement
element = createElement();
assert(element != null);
element!.assignOwner(owner);
});
owner.buildScope(element!, () {
// 核心
element!.mount(null, null);
});
SchedulerBinding.instance!.ensureVisualUpdate();
} else {
element._newWidget = this;
element.markNeedsBuild();
}
return element!;
}
这个方法的核心逻辑也比较清晰,首先来解释下两个参数:
- BuildOwner owner
buildOwner来自WidgetsBinding初始化时实例化的BuildOwner实例
BuildOwner可以理解为widgets管理器,用于跟踪哪些widget需要重建,并且处理适用于整体widget树的其它任务,例如管理widget树的非活动元素列表,以及调试时热重载期间辟谣时触发重新组装
指令。通常由WidgetsBind创建和拥有。
- RenderObjectToWidgetElement? element
element也就是attachRootWidget
传递过来的renderViewElement,其值就是_renderViewElement自己,此时由于调用完appach才赋值,所以首次进来也是null。首次进来为null的时候,就会进入到createElement()逻辑,也就是创建RootElement。
======================================================================
我们继续看Element的mount
方法
@override
void mount(Element? parent, dynamic newSlot) {
super.mount(parent, newSlot);
_renderObject = widget.createRenderObject(this);
}
最后
我的面试经验分享可能不会去罗列太多的具体题目,因为我依然认为面试经验中最宝贵的不是那一个个具体的题目或者具体的答案,而是结束面试时,那一刻你的感受以及多天之后你的回味~
很多人在刚接触这个行业的时候或者是在遇到瓶颈期的时候,总会遇到一些问题,比如学了一段时间感觉没有方向感,不知道该从那里入手去学习,对此我整理了一些资料,需要的可以免费分享给大家
在这里小编分享一份自己收录整理上述技术体系图相关的几十套腾讯、头条、阿里、美团等公司的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。
【Android核心高级技术PDF文档,BAT大厂面试真题解析】
【算法合集】
【延伸Android必备知识点】
【Android部分高级架构视频学习资源】
**Android精讲视频领取学习后更加是如虎添翼!**进军BATJ大厂等(备战)!现在都说互联网寒冬,其实无非就是你上错了车,且穿的少(技能),要是你上对车,自身技术能力够强,公司换掉的代价大,怎么可能会被裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水,赶快领取吧!
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!
裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水,赶快领取吧!
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!