您好!
欢迎来到京东云开发者社区
登录
首页
博文
课程
大赛
工具
用户中心
开源
首页
博文
课程
大赛
工具
开源
更多
用户中心
开发者社区
>
博文
>
Flutter状态管理新的实践
分享
打开微信扫码分享
点击前往QQ分享
点击前往微博分享
点击复制链接
Flutter状态管理新的实践
自猿其说Tech
2022-06-14
IP归属:未知
5342浏览
Flutter
前端
### 1 背景介绍 #### 1.1 声明式ui 声明式UI其实并不是近几年的新技术,但是近几年声明式UI框架非常的火热。单说移动端,跨平台方案有:RN、Flutter。iOS原生有:SwiftUI。android原生有:compose。华为的鸿蒙系统前段时间也发布了基于type-js的ArkUI的beta版。可以看到声明式UI是以后的前端发展趋势。而状态管理是声明式UI框架的重要组成部分。 #### 1.2 声明式UI框架的状态 在移动端之前的命令式UI框架,没有状态的概念。每个控件其实都是无状态的,我们要更新UI需要手动的去set。命令式UI引入状态的概念,状态可以理解为订阅了控件所依赖数据的变化,当一个控件依赖的数据发生变化时,自动刷新UI展示。最大的优势就是可以很方便的做到UI和逻辑的解耦。 ### 2 provider状态管理 #### 2.1 使用方式 定义一个页面如下: ![](//img1.jcloudcs.com/developer.jdcloud.com/fb70026c-6b17-4c0f-9932-939dec79f43520220614143819.png) 实现功能,当点击“按钮”的时候,更新“你好”这个组件 页面部分代码实现(基于StatelessWidget实现): ![](//img1.jcloudcs.com/developer.jdcloud.com/94fcc619-3cb3-4a1a-af4a-bfb8ef09c03020220614143848.jpg) model部分实现: ![](//img1.jcloudcs.com/developer.jdcloud.com/7d64d979-8e13-4619-abe2-397ad8841db620220614143905.png) #### 2.2 问题和不足 点击“按钮”的时候查看页面刷新,发现下表罗列的Widget都执行了刷新操作,使用Selector虽然被包裹的内容没有刷新,但是需要进行校验操作。 ##### 2.2.1 控件刷新 ![](//img1.jcloudcs.com/developer.jdcloud.com/a675b45b-111c-4d52-b423-05cb4e1e183020220614143937.png) ##### 2.2.2 问题分析 1. 使用不太灵活,想要消费事件刷新UI必须有顶层的Provider提供model,在一些复杂场景可能会增加逻辑复杂度 2. 状态刷新,不能实现最小粒度的管理 3. 代码不够简洁 ### 3 新的状态管理方式实践 #### 3.1 使用方式 实现同样的上述页面逻辑,代码如下(同样基于StatelessWidget实现): 首先不需要依赖外部的provider提供Model,任何想要独立刷新的区域使用TosObWidget控件包裹即可,使用比较灵活,我们可以把TosObWidget插入到任何我们想要的位置(包括provider内),代码逻辑比较简洁 ![](//img1.jcloudcs.com/developer.jdcloud.com/9743d0f6-187b-4fc9-8e96-4eaf7d14143320220614144130.png) model实现: model的实现更加简洁,不需要继承ChangeNotifier,所以可以把状态数据定义在任何我们想要的地方,使用.tos扩展属性返回一个包含默认值的RxObj<T>对象,当我们使用set方法更改RxObj<T>的value的时候,通知依赖此对象的TosObWidget区域进行刷新,例:我们点击按钮的时候,_model.textA.value = "你好${_model.i++}",执行后就会刷新依赖textA的TosObWidget(() => Text(_model.textA.value))区域 ![](//img1.jcloudcs.com/developer.jdcloud.com/bc4b3e19-a4a1-4895-afb4-a4ffe26b42ca20220614144141.jpg) 查看刷新状态(与provider对比): ![](//img1.jcloudcs.com/developer.jdcloud.com/6d124089-6fb1-42bd-8162-395c465f687220220614144153.png) 对比发现TosObWidget这种方式,只有依赖的数据发生变化的TosObWidget才会更新状态,可以实现状态刷新粒度最小化,提高性能 #### 3.2 设计思路 ##### 3.2.1 TosObWidget ![](//img1.jcloudcs.com/developer.jdcloud.com/1e0073d0-3fde-4b38-a950-f223f629c57c20220614144217.png) 首先是使用入口,定义一个TosObWidget控件,入参为build函数,返回widget,每个TosObWidget就是一个可独立进行状态刷新的区域 ![](//img1.jcloudcs.com/developer.jdcloud.com/8aa4784e-0864-4a85-bdb1-f9a36d4606b320220614153119.jpg) TosObWidget控件的实现如下: ![](//img1.jcloudcs.com/developer.jdcloud.com/893bf906-f02c-4f15-a616-6fe38aa9023120220614153130.jpg) TosObWidget的build函数为重载的其父类_ObzWidget的build函数,最终会被_ObzWidget的_ObzState调用,_ObzWidget的实现如下: ![](//img1.jcloudcs.com/developer.jdcloud.com/103cc9ce-7c6c-45c9-aa5e-2aeeb0b8a28a20220614153145.jpg) 接下来查看_ObzState的实现,主要逻辑都在这个类进行实现,这里贴出所有的代码(注意框起来的逻辑): ![](//img1.jcloudcs.com/developer.jdcloud.com/6e1fad1c-318b-4b07-aba1-d1985b75a80820220614153156.jpg) ![](//img1.jcloudcs.com/developer.jdcloud.com/6526a8cc-2b82-4364-818e-66f1106257bd20220614153206.jpg) ##### 3.2.2 TosObWidget逻辑分析 1. 首先_ObzState依赖一个RxObserver _observer变量 2. RxObserver _observer这个 变量持有了_updateUI()这个方法,最终会通过这个方法刷新TosOBWidget的状态 3. 当TosObWidget执行build的时候,会通过一个静态变量RxObserver.proxy把_observer共享出去 4. 这样TosObWidget包裹的内容,使用RxObj的getValue的时候会拿到被共享的_observer,这时建立RxObj和TosObWidget的联系 5. 联系建立后,重置共享变量RxObserver.proxy 6. 这样在RxObj的value执行set方法时,会调用到与其绑定的TosObWidget的_updateUI()这个函数 ##### 3.2.3 RxObj的实现 ![](//img1.jcloudcs.com/developer.jdcloud.com/58080636-2283-419b-a2a0-499d3b8f9e8720220614153231.png) 如下贴出RxObj的value的get和set函数: 1. 当执行RxObj的value的get方法时,代码如下,拿到 RxObserver的静态成员变量proxy,类型为RxObserver(即为上一步TosObWidget共享出来的_observer) 2. 判断RxObserver.proxy不为空,且没有被添加到_observers列表( List<RxObserver> _observers),则添加 3. 当执行RxObj的value的set方法时,校验value是否与当前的value值相同,且判断是否是首次创建(首次创建不会执行状态刷新) 4. 校验完成后则赋值执行refresh()函数,更新TosObWidget的状态 ![](//img1.jcloudcs.com/developer.jdcloud.com/71311bec-de48-43a1-85fb-915aa1052acf20220614153300.jpg) refresh()函数的实现如下: observer.update()函数即为执行与Rxobj关联的TosObWidget的_updateUI()函数: ![](//img1.jcloudcs.com/developer.jdcloud.com/b8687110-fb2e-40ce-bc57-f5730b5a509e20220614153316.jpg) 看下RxObserver的实现: 注意框起来的逻辑,update函数即上面_ObzState的_updateUI()函数的引用 ![](//img1.jcloudcs.com/developer.jdcloud.com/e874dd38-c99a-44ef-a05f-c5b3e676dbf520220614153434.jpg) 至此整个实现流程已经贯通了,接下来看下如何使用: 1)通过.tos扩展属性定义RxObj变量: ![](//img1.jcloudcs.com/developer.jdcloud.com/a47b1fd9-1e45-4281-9eb0-4d1bd2c5fce120220614153449.jpg) 2).tos扩展属性的实现如下: ![](//img1.jcloudcs.com/developer.jdcloud.com/22ab9105-6275-4885-aa7d-ff48b47df03020220614153503.jpg) 3)如果要创建一个默认值为空的,RxObj实例,使用如下方式: ![](//img1.jcloudcs.com/developer.jdcloud.com/9dc07c16-5e48-459b-857e-4c34e63cb7b120220614153520.jpg) 此时如果我们使用RxObj的setValue方法,就会刷新依赖它的所有TosObWidget控件,如果有些情况下,没有调用setValue方法,但是需要刷新状态,可手动调用refresh()方法,实现如下: ![](//img1.jcloudcs.com/developer.jdcloud.com/37d948c0-c64c-42a5-ad93-576222e77a1b20220614153531.jpg) 至此,就完成了TosObWidget控件的状态刷新 ### 4 总结 注:基于本文示例的功能逻辑进行对比 ![](//img1.jcloudcs.com/developer.jdcloud.com/f1434e15-7189-4523-ae8d-154cca87985420220614153546.png) ------------ ###### 自猿其说Tech-JDL京东物流技术与数据智能部 ###### 作者:张俊飞
原创文章,需联系作者,授权转载
上一篇:Quartz核心原理之架构及基本元素介绍(一)
下一篇:MYSQL中JSON类型介绍
相关文章
【技术干货】企业级扫描平台EOS关于JS扫描落地与实践!
Flutter异步编程中Completer的使用
聊一聊多线程不得不知的Future(一)
自猿其说Tech
文章数
426
阅读量
2163641
作者其他文章
01
深入JDK中的Optional
本文将从Optional所解决的问题开始,逐层解剖,由浅入深,文中会出现Optioanl方法之间的对比,实践,误用情况分析,优缺点等。与大家一起,对这项Java8中的新特性,进行理解和深入。
01
Taro小程序跨端开发入门实战
为了让小程序开发更简单,更高效,我们采用 Taro 作为首选框架,我们将使用 Taro 的实践经验整理了出来,主要内容围绕着什么是 Taro,为什么用 Taro,以及 Taro 如何使用(正确使用的姿势),还有 Taro 背后的一些设计思想来进行展开,让大家能够对 Taro 有个完整的认识。
01
Flutter For Web实践
Flutter For Web 已经发布一年多时间,它的发布意味着我们可以真正地使用一套代码、一套资源部署整个大前端系统(包括:iOS、Android、Web)。渠道研发组经过一段时间的探索,使用Flutter For Web技术开发了移动端可视化编程平台—Flutter乐高,在这里希望和大家分享下使用Flutter For Web实践过程和踩坑实践
01
配运基础数据缓存瘦身实践
在基础数据的常规能力当中,数据的存取是最基础也是最重要的能力,为了整体提高数据的读取能力,缓存技术在基础数据的场景中得到了广泛的使用,下面会重点展示一下配运组近期针对数据缓存做的瘦身实践。
自猿其说Tech
文章数
426
阅读量
2163641
作者其他文章
01
深入JDK中的Optional
01
Taro小程序跨端开发入门实战
01
Flutter For Web实践
01
配运基础数据缓存瘦身实践
添加企业微信
获取1V1专业服务
扫码关注
京东云开发者公众号