前言

一个永恒的主题,“状态(State)管理”,无论是在React/Vue(两者都是支持响应式编程的Web开发框架)还是Flutter中,他们讨论的问题和解决的思想都是一致的。

一个问题,StatefulWidget的状态应该被谁管理?Widget本身?父Widget?都会?还是另一个对象?答案是取决于实际情况!以下是管理状态的最常见的方法

  1. Widget管理自己的状态。
  2. Widget管理子Widget状态。
  3. 混合管理(父Widget和子Widget都管理状态)。

如何决定使用哪种管理方法?下面是官方给出的一些原则

  1. 如果状态是用户数据,如复选框的选中状态、滑块的位置,则该状态最好由父Widget管理。
  2. 如果状态是有关界面外观效果的,例如颜色、动画,那么状态最好由Widget本身来管理。
  3. 如果某一个状态是不同Widget共享的则最好由它们共同的父Widget管理。

三个例子

下面通过三个例子来分别说明这三种管理方式。

Widget管理自身状态

_TapBoxAState 类:

  • TapBoxA。
  • 定义_active:确定盒子的当前颜色的布尔值。
  • 定义_handleTap()函数,该函数在点击该盒子时更新_active,并调用setState()更新UI。
  • 实现widget的所有交互式行为。
class TapBoxA extends StatefulWidget {
TapBoxA({Key key}) : super(key: key); @override
_TapBoxAState createState() => _TapBoxAState();
} class _TapBoxAState extends State<TapBoxA> {
bool _active = false; void _handleTap() {
setState(() {
_active = !_active;
});
} Widget build(BuildContext context) {
// 使用GestureDetector来识别点击事件
return GestureDetector(
onTap: _handleTap,
child: Container(
child: Center(
child: Text(
_active ? 'Active' : 'Inactive',
style: TextStyle(fontSize: 32.0, color: Colors.white),
),
),
width: 200.0,
height: 200.0,
decoration: BoxDecoration(
color: _active ? Colors.lightGreen[700] : Colors.grey[700],
),
),
);
}
}

父Widget管理子Widget的状态

对于父Widget来说,管理状态并告诉其子Widget何时更新通常是比较好的方式。

例如,IconButton是一个图标按钮,但它是一个无状态的Widget,因为我们认为父Widget需要知道该按钮是否被点击来采取相应的处理。

ParentWidgetState 类:

  1. 为TapBoxB 管理_active状态。
  2. 实现_handleTaPBoxChanged(),当盒子被点击时调用的方法。
  3. 当状态改变时,调用setState()更新UI。

    TapBoxB 类:
  4. 继承StatelessWidget类,因为所有状态都由其父组件处理。
  5. 当检测到点击时,它会通知父组件。

TaPBoxB通过回调将其状态导出到其父组件,状态由父组件管理,因此它的父组件为StatefulWidget。

但是由于TapBoxB不管理任何状态,所以TapBoxB为StatelessWidget。

class ParentWidgetB extends StatefulWidget {
@override
_ParentWidgetBState createState() => _ParentWidgetBState();
} class _ParentWidgetBState extends State<ParentWidgetB> {
bool _active = false; void _handleTaPBoxChanged(bool newValue) {
setState(() {
_active = newValue;
});
} @override
Widget build(BuildContext context) {
return Container(
child: TapBoxB(
active: _active,
onChanged: _handleTaPBoxChanged,
),
);
} } class TapBoxB extends StatelessWidget {
TapBoxB({Key key, this.active: false, @required this.onChanged}): super(key: key); final bool active;
final ValueChanged<bool> onChanged; void _handleTap() {
onChanged(!active);
} @override
Widget build(BuildContext context) {
return GestureDetector(
onTap: _handleTap,
child: Container(
child: Center(
child: Text(
active ? 'Active' : 'Inactive',
style: TextStyle(fontSize: 32.0, color: Colors.white),
),
),
width: 200.0,
height: 200.0,
decoration: BoxDecoration(
color: active ? Colors.lightGreen[400] : Colors.grey[400],
),
),
);
}
}

混合状态管理

TapBoxC示例中,手指按下时,盒子的周围会出现一个深绿色的边框,抬起时,边框消失。

点击完成后,盒子的颜色改变。TapBoxC将其_active状态导出到其父组件中,但在内部管理其_highlight状态。

这个例子有两个状态对象_ParentWidgetState和_TaPBoxCState。

_ParentWidgetStateC类:

  1. 管理_active 状态。
  2. 实现 _handleTaPBoxChanged() ,当盒子被点击时调用。
  3. 当点击盒子并且_active状态改变时调用setState()更新UI。

_TapBoxCState 对象:

  1. 管理_highlight 状态。
  2. GestureDetector监听所有tap事件。当用户点下时,它添加高亮(深绿色边框);当用户释放时,会移除高亮。
  3. 当按下、抬起、或者取消点击时更新_highlight状态,调用setState()更新UI。
  4. 当点击时,将状态的改变传递给父组件。
class ParentWidgetC extends StatefulWidget {
@override
_ParentWidgetCState createState() => _ParentWidgetCState();
} class _ParentWidgetCState extends State<ParentWidgetC> {
bool _active = false; void _handleTapBoxChanged(bool newValue) {
setState(() {
_active = newValue;
});
} @override
Widget build(BuildContext context) {
return Container(
child: TapBoxC(active: _active, onChanged: _handleTapBoxChanged,),
);
}
} class TapBoxC extends StatefulWidget {
TapBoxC({Key key, this.active: false, @required this.onChanged}): super(key: key); final bool active;
final ValueChanged<bool> onChanged; _TapBoxCState createState() => _TapBoxCState();
} class _TapBoxCState extends State<TapBoxC> {
bool _highlight = false; void _handleTapDown(TapDownDetails details) {
setState(() {
_highlight = true;
});
} void _handleTapUp(TapUpDetails details) {
setState(() {
_highlight = false;
});
} void _handleTapCancel() {
setState(() {
_highlight = false;
});
} void _handleTap() {
widget.onChanged(!widget.active);
} @override
Widget build(BuildContext context) {
// 在按下时添加绿色边框,当抬起时,取消高亮
return GestureDetector(
onTapDown: _handleTapDown, // 处理按下事件
onTapUp: _handleTapUp, // 处理抬起事件
onTap: _handleTap,
onTapCancel: _handleTapCancel,
child: Container(
child: Center(
child: Text(
widget.active ? 'Active' : 'Inactive',
style: TextStyle(fontSize: 32.0, color: Colors.white),
),
),
width: 200.0,
height: 200.0,
decoration: BoxDecoration(
color: widget.active ? Colors.lightGreen[200] : Colors.grey[200],
border: _highlight ? Border.all(color: Colors.teal[700], width: 10.0,) : null,
),
),
);
}
}

全局状态管理

当应用中需要一些跨组件(包括跨路由)的状态需要同步时,上面三种管理方式就很难胜任了。

有一个设置页,里面可以设置应用的语言,为了让设置实时生效,期望在语言状态发生改变时,APP中依赖应用语言的组件能够重新build一下,但这些依赖应用语言的组件和设置页并不在一起。

而全局状态管理则可以处理这种相距较远的组件之间的通信,目前有两种方法:

  1. 实现一个全局的事件总线,将语言状态改变为一个事件,然后在App中依赖应用语言的组件的initState方法中订阅语言改变的事件。

    当用户设置页切换语言后,发布语言改变事件,而订阅了此事件的组件就会收到通知,收到通知后调用setState()重新build一下即可。
  2. 使用一些专门用于状态管理的包,如Provider、Redux等。

Flutter 基础组件:状态管理的更多相关文章

  1. Flutter 基础组件:单选框和复选框

    前言 Material组件库中提供了Material风格的单选开关Switch和复选框Checkbox,虽然它们都是继承自StatefulWidget,但它们本身不会保存当前选中状态,选中状态都是由父 ...

  2. Flutter 基础组件:输入框和表单

    前言 Material组件库中提供了输入框组件TextField和表单组件Form. 输入框TextField 接口描述 const TextField({ Key key, // 编辑框的控制器,通 ...

  3. Flutter 基础组件:按钮

    前言 Material组件库中提供了多种按钮组件如RaisedButton.FlatButton.OutlineButton等,它们都是直接或间接对RawMaterialButton组件的包装定制,所 ...

  4. Flutter 基础组件:Widget简介

    概念 在Flutter中几乎所有的对象都是一个Widget.与原生开发中"控件"不同的是,Flutter中的Widget的概念更广泛,它不仅可以表示UI元素,也可以表示一些功能性的 ...

  5. Flutter 基础组件:进度指示器

    前言 Material 组件库中提供了两种进度指示器:LinearProgressIndicator和CircularProgressIndicator,它们都可以同时用于精确的进度指示和模糊的进度指 ...

  6. Flutter 基础组件:图片和Icon

    前言 Flutter中,可以通过Image组件来加载并显示图片,Image的数据源可以是asset.文件.内存以及网络. ImageProvider 是一个抽象类,主要定义了图片数据获取的接口load ...

  7. Flutter 基础组件:文本、字体样式

    // 文本.字体样式 import 'package:flutter/material.dart'; class TextFontStyle extends StatelessWidget { // ...

  8. flutter 基础组件

    TextWidget class TextWidget extends StatelessWidget { final TextStyle _textStyle = TextStyle( fontSi ...

  9. JSP | 基础 | JSP状态管理 | Cookie

    Cookie : 是web服务器保存在客户端的一系列文本信息. Cookie的作用: 1.对特定的对象的追踪 2. 3. JSP中创建Cookie以及使用 创建Cookie对象 写入Cookie对象 ...

随机推荐

  1. Deep Learning with Differential Privacy

    原文链接:Deep Learning with Differential Privacy abstract:新的机器学习算法,差分隐私框架下隐私成本的改良分析,使用非凸目标训练深度神经网络. 数学中最 ...

  2. REHの收藏列表

    搬运自本人的AcWing,所以那里的文章会挺多. 友链(同类文章) :bztMinamoto 世外明月 mlystdcall 新人手册:AcWing入门使用指南 前言 有看到好文欢迎推荐(毛遂自荐也可 ...

  3. SQL实现_同时在线人数

    原始数据表结构如下: user_id login_time logout_time 12 2020-12-10 20:45:18 2020-12-10 21:45:18 只说下实现思路,SQL不太难, ...

  4. NameSilo的DDNS动态域名解析

    用Java写的,一个实时检测IP变化并更新DNS状态的工具,适用于在NameSilo购买的域名,如果你的域名是在其他商家购买的,修改为你自己的api就行.代码我放github了,地址: https:/ ...

  5. 微信开发中,不同手机系统遇到的bug(不定时更新)

    Ios系统 1.body上绑定click事件失效. 解决:body标签下面,用个div,当做包裹所有内容的大容器.给这个div,绑定click事件. 2.不支持 YYYY-MM-DD 的时间格式. 用 ...

  6. CIBN手机电视8.3.2永久VIP

    一款互联网电视的手机客户端.可以观看最新的电影和电视剧,还会为你推荐人气热门电影,让你不会错过每一部精彩的大片,以去除app内的所有可见广告,解锁VIP特权,无需登录直接使用! 下载地址:https: ...

  7. DVWA各等级命令注入漏洞

    漏洞描述 在web程序中,因为业务功能需求要通过web前端传递参数到后台服务器上执行,由于开发人员没有对输入进行严格过滤,导致攻击者可以构造一些额外的"带有非法目的的"命令,欺骗后 ...

  8. monkey在指定的activity里面运行

    下载包地址:链接: https://pan.baidu.com/s/1Wk2eOj3saZx71Mx6pT2L4Q 提取码: gupa 运行方式:步骤1: 将工具下载下来放到本地目录下,解压步骤2:配 ...

  9. 轮廓检测论文解读 | Richer Convolutional Features for Edge Detection | CVPR | 2017

    有什么问题可以加作者微信讨论,cyx645016617 上千人的粉丝群已经成立,氛围超好.为大家提供一个遇到问题有可能得到答案的平台. 0 概述 论文名称:"Richer Convoluti ...

  10. 第 16 章 【硬核!】 垃圾回收相关 GC细讲

    第 16 章 垃圾回收相关概念 1.System.gc() 的理解 1.1.System.gc() 方法 System.gc() 方法 在默认情况下,通过System.gc()者Runtime.get ...