Flutter开发中的一些Tips(四)

在这里插入图片描述
最近接手了一个flutter项目,整体感觉代码质量不高,感觉有些是初学者容易犯的问题。几年前写的前三篇,我是站在我自己开发遇到问题的角度,这篇是站在别人遇到问题的角度,算是一种补充。下面我整理一下遇到的小问题,大家可以当作开发中的Tips。

1.使用“平替”Widget

Spacer

有时候在Row或者Column中需要占位会有如下写法:

Expanded(child: Container())
/// 或
const Expanded(child: SizedBox())

上面两种相比我更推荐第二种,因为可以加const。另外第一种方式,因为Container没有child,所以它的大小会撑满父布局。在Stack中使用需要注意,别问我为什么突然这么说,因为我就遇到这样写的,然后出了问题。。。

当然Flutter还有自带的Spacer,源码如下:

class Spacer extends StatelessWidget {const Spacer({super.key, this.flex = 1}): assert(flex > 0);final int flex;Widget build(BuildContext context) {return Expanded(flex: flex,child: const SizedBox.shrink(),);}
}

等于是上面第二种方法封装了一层,使用起来也更方便,const Spacer()就搞定了。

Nil

有时会根据一个条件来判断显示什么widget。当我们无法返回null时,我们会返回类似const SizedBox()的东西。

这很好,但自从SizedBox创建RenderObject,它有一些性能影响。RenderObject位于渲染树中,并在上面执行一些计算,即使它在屏幕上没有绘制任何东西。
所以我们可以有一个不创建RenderObject的Widget,同时仍然有效。Nil小部件是它的实现。它只创建一个Element,在构建时什么也不做。

代码如下:

/// A widget which is not in the layout and does nothing.
/// It is useful when you have to return a widget and can't return null.
class Nil extends Widget {/// Creates a [Nil] widget.const Nil({super.key});Element createElement() => _NilElement(this);
}class _NilElement extends Element {_NilElement(Nil super.widget);void mount(Element? parent, dynamic newSlot) {assert(parent is! MultiChildRenderObjectElement, """You are using Nil under a MultiChildRenderObjectElement.This suggests a possibility that the Nil is not needed or is being used improperly.Make sure it can't be replaced with an inline conditional oromission of the target widget from a list.""");super.mount(parent, newSlot);}bool get debugDoingBuild => false;void performRebuild() {super.performRebuild();}
}

目前Nil不支持RowColumn这类(MultiChildRenderObjectElement)多个子节点的组件。

ColoredBox/SizedBox/DecoratedBox等

如果只是设置背景色,可以完全使用ColoredBox替代Container;同理如果只是设置大小,可以使用SizedBox;只设置边框圆角渐变这类,可以使用DecoratedBox。类似的还有PaddingTransform等。如果你需要以上的多种组合,这个时候推荐Container ,因为它就是以上实现的封装。

为什么如此?Container是一个比ColoredBox等更重的小部件,里面的实现不都是我们需要的。同时也是为了尽可能的使用const声明。const的问题,我在很早之前的Flutter性能优化实践 —— UI篇中就有说明过:

有人测试一个页面上构建1000个重复图标,结果使用const构造函数的,FPS大约高8.4%,内存使用量降低约20%。
当然,实际一个页面上有1000个Widget也不现实。其实说这个点的原因也是希望大家能养成一个好习惯。

如果你的linter启用了use_colored_box或是sized_box_for_whitespace,当你有上述“错误”写法时,编译器也会给你相应的警告。

2.嵌套

Flutter的地狱嵌套一直被许多人诟病,觉得代码看起来很乱。但是实际上很好解决这个问题。

  • Widget封装。封装公共组件,实现拆分。
  • 善用ThemeData,可以避免重复设置属性。
  • 可以使用flutter_constraintlayout 库,它和 Android 下的 ConstraintLayout ,iOS 下的 AutoLayout 相似。可以将“阶梯状”的代码变得“扁平”。我自己在日常开发中优先就使用的它,可以让我无脑布局。
  • 换一种嵌套方式。大家可以看下Flutter的源码,学习一下代码风格。例如Container 源码:
 Widget build(BuildContext context) {Widget? current = child;if (child == null && (constraints == null || !constraints!.isTight)) {current = LimitedBox(maxWidth: 0.0,maxHeight: 0.0,child: ConstrainedBox(constraints: const BoxConstraints.expand()),);} else if (alignment != null) {current = Align(alignment: alignment!, child: current);}final EdgeInsetsGeometry? effectivePadding = _paddingIncludingDecoration;if (effectivePadding != null) {current = Padding(padding: effectivePadding, child: current);}if (color != null) {current = ColoredBox(color: color!, child: current);}...if (margin != null) {current = Padding(padding: margin!, child: current);}if (transform != null) {current = Transform(transform: transform!, alignment: transformAlignment, child: current);}return current!;}

可以看到是一种从上到下嵌套的方式。比如我们写一个控件,可以先写内容,然后再套大小间距样式,再套点击事件这种。这样代码就会清晰很多。下面是一个简单的例子:

 Widget build(BuildContext context) {Widget child = Row(children: <Widget>[Text(title),const Spacer(),...],);child = Container(margin: const EdgeInsets.only(left: 15.0),padding: const EdgeInsets.fromLTRB(0, 15.0, 15.0, 15.0),constraints: const BoxConstraints(minHeight: 50.0,),width: double.infinity,decoration: BoxDecoration(border: Border(bottom: Divider.createBorderSide(context, width: 0.6),),),child: child,);return InkWell(onTap: onTap,child: child,);}

3.Linter规则

使用linter可以帮助我们识别Dart代码中可能存在的问题,当有“问题”时,编译器会有警告,同时里面会有文档链接详细解释说明。另一方面在团队开发中也可以统一代码规范。

一般新建项目后,会自动添加flutter_lints库。这个里面有lints库中有关dart的core规则集,和recommended规则集,加上flutter_lints中的flutter规则集。我数了一下大概八十多条规则,但我觉得还远远不够。比如上面提到有关ColoredBoxuse_colored_box这条就没有。

所以我个人推荐使用Flutter项目中的analysis_options.yaml内容来替换flutter_lints,然后可以基于此规范再做调整。目前Flutter项目中的analysis_options.yaml大致有两百条规则左右。更多的规则可以给到更多代码建议,同时统一标准,长远看是有利于团队的。

比如最近我看见analysis_options.yaml中新增了一条规则strict-inference,它是当无法确定静态类型时,类型推断永远不会选择动态类型。
在这里插入图片描述
加上这条后,我就发现了我之前的“错误”写法。

在这里插入图片描述
就是因为我没有声明方法的返回类型,那么以前编译器就会推断为dynamic。所以修改很简单,前面加上void就好了。

其实看下flutter源码,你就会发现都是很标准的写法:

在这里插入图片描述

所以对于初学者我更是建议学的时候就直接上高标准,学的时候就严要求自己。否则“不良”的代码习惯一旦养成,后面再改就比较痛苦了。

4.SafeArea

SafeArea内部通过Padding添加间距来避免我们的widget和状态栏/刘海/底部的安全区域重叠。但是使用时需要注意,默认情况下SafeArea的上下左右四个属性都是true。所以如果包裹的控件在上方显示,注意将bottom改为false,否则如果设备bottom不为0时,包裹的控件下方会有高度占用,影响下方排列的widget。

另外就是横屏使用时,注意考虑左右方向。

5.Mixin

Mixin(混入)是Dart语言的一个特性。有别于extends的单继承,它是一种在多类层次结构中复用代码的一种方式。我个人认为是必须要掌握的,使用起来方便,也是封装功能的首选。我们平时使用的SingleTickerProviderStateMixinAutomaticKeepAliveClientMixin都是利用Mixin实现的。

下面我举个小例子说明一下用法。比如平时会使用各种Controller来监听动画,滑动之类的,页面销毁时我们要将这些Controller释放掉。就会有类似下面的代码:

class TestPageState extends State<TestPage> {final TextEditingController _controller = TextEditingController();final FocusNode _nodeText = FocusNode();void initState() {_controller.addListener(callback);super.initState();}void dispose() {_controller.removeListener(callback);_controller.dispose();_nodeText.dispose();super.dispose();}}

如果一个页面上有许多Controller时,会写许多这样的代码。说实话挺麻烦的,一个不留神可能还会漏掉。那我们完全可以把这些相似的操作封装起来。

mixin ChangeNotifierMixin<T extends StatefulWidget> on State<T> {Map<ChangeNotifier?, List<VoidCallback>?>? _map;Map<ChangeNotifier?, List<VoidCallback>?>? changeNotifier();void initState() {_map = changeNotifier();/// 遍历数据,如果callbacks不为空则添加监听_map?.forEach((changeNotifier, callbacks) { if (callbacks != null && callbacks.isNotEmpty) {void addListener(VoidCallback callback) {changeNotifier?.addListener(callback);}callbacks.forEach(addListener);}});super.initState();}void dispose() {_map?.forEach((changeNotifier, callbacks) {if (callbacks != null && callbacks.isNotEmpty) {void removeListener(VoidCallback callback) {changeNotifier?.removeListener(callback);}callbacks.forEach(removeListener);}changeNotifier?.dispose();});super.dispose();}
}

在页面执行initStatedispose方法时,就会自动执行Mixin中的这两个方法,就像是方法“混入”其中一样。

修改后,使用方法:

class TestPageState extends State<TestPage> with ChangeNotifierMixin<TestPage> {final TextEditingController _controller = TextEditingController();final FocusNode _nodeText = FocusNode();Map<ChangeNotifier, List<VoidCallback>?>? changeNotifier() {return {_controller: [callback],_nodeText: null,};}}

这样处理后,代码是不是简洁了许多。而且使用起来只需要添加ChangeNotifierMixin实现changeNotifier方法就行了。如果用抽象去封装,效果可以达到一样,但是无法多个继承,都放在一个base下会越来越臃肿。Mixin有种即插即用的感觉,既可以像继承一样方便,又可以像接口一样实现多个,非常好用。

同理,Dart的扩展方法,枚举扩展这些语法特性,都是需要掌握的。


本篇到此结束,同时推荐阅读前三篇。后面有补充内容也会更新到这里。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

中国专利转让数据集(1985-2021年)

专利转让数据追踪和记录专利从一个实体转移到另一个实体的过程。这些数据不仅包括参与转让的申请人和受让人的身份信息&#xff0c;如名字和地址&#xff0c;还涵盖了转让的具体法律细节&#xff0c;包括转让执行日、转让次数、法律状态变更&#xff0c;以及转让登记的相关信息…

在Spring Boot中使用MyBatis访问数据库

MyBatis&#xff0c;这个对各位使用Java开发的开发者来说还是蛮重要的&#xff0c;我相信诸位在企业开发项目的时候&#xff0c;大多数采用的是Mybatis。使用MyBatis帮助我们解决各种问题&#xff0c;实际上这篇文章&#xff0c;基本上默认为可以跳过的一篇&#xff0c;但是为了…

Linux服务器从零开始训练 RT-DETR 改进项目 (Ultralytics) 教程,改进RTDETR算法(包括使用训练、验证、推理教程)

手把手从零开始训练 RT-DETR 改进项目 (Ultralytics版本) 教程,改进RTDETR算法 本文以Linux服务器为例:从零开始使用Linux训练 RT-DETR 算法项目 《芒果剑指 RT-DETR 目标检测算法 改进》 适用于芒果专栏改进RT-DETR算法 文章目录 百度 RT-DETR 算法介绍改进网络代码汇总第…

arcgis基础篇--实验

一、绘制带空洞的面要素 方法一&#xff1a;先绘制出一个面区域&#xff0c;然后在面上再绘制一个面区域代表面洞&#xff0c;两者位于同一个图层内&#xff0c;选中代表面洞的区域&#xff0c;选择【编辑器】-【裁剪】工具&#xff0c;将面裁剪出一个洞&#xff0c;随后删除代…

openinstall携手途虎养车,赋能汽车服务数字化

近日&#xff0c;openinstall与中国领先的一站式汽车服务平台途虎养车再次续约&#xff0c;双方将开启第三年合作。过去两年&#xff0c;途虎在建设线上线下一体化数字平台的过程中&#xff0c;深度结合openinstall传参归因与渠道统计技术&#xff0c;打造出了一套高效的渠道来…

GZ038 物联网应用开发赛题第4套

2023年全国职业院校技能大赛 高职组 物联网应用开发 任 务 书 &#xff08;第4套卷&#xff09; 工位号&#xff1a;______________ 第一部分 竞赛须知 一、竞赛要求 1、正确使用工具&#xff0c;操作安全规范&#xff1b; 2、竞赛过程中如有异议&#xff0c;可向现场考评…

Hadoop学习总结(使用Java API操作HDFS)

使用Java API操作HDFS&#xff0c;是在安装和配置Maven、IDEA中配置Maven成功情况下进行的&#xff0c;如果Maven安装和配置不完全将不能进行Java API操作HDFS。 由于Hadoop是使用Java语言编写的&#xff0c;因此可以使用Java API操作Hadoop文件系统。使用HDFS提供的Java API构…

C语言进阶

数组 在基础篇说过&#xff0c;数组实际上是构造类型之一&#xff0c;是连续存放的。 一维数组 定义 定义格式&#xff1a;[存储类型] 数据类型 数组名标识符[下标]; 下面分模块来介绍一下数组的定义部分的内容。 1、初始化和元素引用&#xff1a; 可以看到数组是连续存储…

Linux--gcc/g++

一、gcc/g是什么 gcc的全称是GNU Compiler Collection&#xff0c;它是一个能够编译多种语言的编译器。最开始gcc是作为C语言的编译器&#xff08;GNU C Compiler&#xff09;&#xff0c;现在除了c语言&#xff0c;还支持C、java、Pascal等语言。gcc支持多种硬件平台 二、gc…

数据结构 队列(C语言实现)

目录 1.队列的概念及结构2.队列的代码实现 正文开始前给大家推荐个网站&#xff0c;前些天发现了一个巨牛的 人工智能学习网站&#xff0c; 通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。 点击跳转到网站。 1.队列的概念及结构 队列&#xff1a;只允许在…

Java Web——TomcatWeb服务器

目录 1. 服务器概述 1.1. 服务器硬件 1.2. 服务器软件 2. Web服务器 2.1. Tomcat服务器 2.2. 简单的Web服务器使用 1. 服务器概述 服务器指的是网络环境下为客户机提供某种服务的专用计算机&#xff0c;服务器安装有网络操作系统和各种服务器的应用系统服务器的具有高速…

【2011年数据结构真题】

41题 41题解答&#xff1a; &#xff08;1&#xff09;图 G 的邻接矩阵 A 如下所示&#xff1a; 由题意得&#xff0c;A为上三角矩阵&#xff0c;在上三角矩阵A[6][6]中&#xff0c;第1行至第5行主对角线上方的元素个数分别为5, 4, 3, 2, 1 用 “ 平移” 的思想&#xff0c;…

vmware workstation 与 device/credential guard 不兼容

VM虚拟机报错 vmware虚拟机启动时报错&#xff1a;vmware workstation 与 device/credential guard 不兼容&#xff1a; 系统是win10专业版&#xff0c;导致报错原因最终发现是安装了docker&#xff0c;docker自带下载虚拟机Hyper-V&#xff0c;而导致vmware workstation 与 …

FPGA高端项目:图像采集+GTX+UDP架构,高速接口以太网视频传输,提供2套工程源码加QT上位机源码和技术支持

目录 1、前言免责声明本项目特点 2、相关方案推荐我这里已有的 GT 高速接口解决方案我这里已有的以太网方案 3、设计思路框架设计框图视频源选择OV5640摄像头配置及采集动态彩条视频数据组包GTX 全网最细解读GTX 基本结构GTX 发送和接收处理流程GTX 的参考时钟GTX 发送接口GTX …

MSF图形化工具Viper快速安装

简介 Viper(炫彩蛇)是一款图形化内网渗透工具,将内网渗透过程中常用的战术及技术进行模块化及武器化. Viper(炫彩蛇)集成杀软绕过,内网隧道,文件管理,命令行等基础功能. Viper(炫彩蛇)当前已集成70个模块,覆盖初始访问/持久化/权限提升/防御绕过/凭证访问/信息收集/横向移动等…

Python的基础语句大全

以下是Python的基础语句大全&#xff1a; 变量定义语句&#xff1a; var_name var_value输出语句&#xff1a; print(var_name)输入语句&#xff1a; var_name input()条件语句&#xff1a; if condition:// do something if condition is True elif condition:// do somethi…

STM32F407: CMSIS-DSP库的移植(基于库文件)

目录 1. 源码下载 2. DSP库源码简介 3.基于库的移植(DSP库的使用) 3.1 实验1 3.2 实验2 4. 使用V6版本的编译器进行编译 上一篇&#xff1a;STM32F407-Discovery的硬件FPU-CSDN博客 1. 源码下载 Github地址&#xff1a;GitHub - ARM-software/CMSIS_5: CMSIS Version 5…

Golang 字符串处理汇总

1. 统计字符串长度&#xff1a;len(str) len(str) 函数用于统计字符串的长度&#xff0c;按字节进行统计&#xff0c;且该函数属于内置函数也不用导包&#xff0c;直接用就行&#xff0c;示例如下&#xff1a; //统计字符串的长度,按字节进行统计: str : "golang你好&qu…

【开源】基于Vue.js的智能停车场管理系统的设计和实现

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、研究内容A. 车主端功能B. 停车工作人员功能C. 系统管理员功能1. 停车位模块2. 车辆模块3. 停车记录模块4. IC卡模块5. IC卡挂失模块 三、界面展示3.1 登录注册3.2 车辆模块3.3 停车位模块3.4 停车数据模块3.5 IC卡档案模块3.6 IC卡挂…

Linux中字符设备的打开、写入

一个内核模块应该由以下几部分组成。 第一部分&#xff0c;头文件部分。一般的内核模块&#xff0c;都需要 include 下面两个头文件&#xff1a; #include <linux/module.h> #include <linux/init.h> 第二部分&#xff0c;定义一些函数&#xff0c;用于处理内核…