Flutter Widget:State 状态管理

响应式的编程框架永恒的主题------"状态(State)管理"

无论是在 React/Vue/Flutter 中讨论的问题和解决的思想都是一致的。

StatefulWidget的状态应该被谁管理?Widget本身?父 Widget ?都会?还是另一个对象?

下面是官方给出的一些原则:

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

以下是管理状态的最常见的方法:

  • Widget 管理自身的状态。
  • Widget 管理子 Widget 状态。
  • 混合管理(父 Widget 和子 Widget 都管理状态)。

接下来,我们将通过例子说明管理状态的不同方式:创建一个盒子,当点击它时,盒子背景会在绿色与灰色之间切换。状态 _active确定颜色:绿色为true ,灰色为false。如图所示:


Widget管理自身状态

我们实现一个TapboxA,对应的状态类_TapboxAState :

Dart 复制代码
class TapboxA extends StatefulWidget {
  TapboxA({Key? key}) : super(key: key);
  @override
  _TapboxAState createState() => _TapboxAState();
}
class _TapboxAState extends State<TapboxA> {
  bool _active = false;//确定盒子的当前颜色的布尔值。
  void _handleTap() {//更新_active,并调用setState()更新UI。
    setState(() { _active = !_active;});
  }
  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[700] : Colors.grey[600],),
      ),);
  }
}

Widget管理子widget状态

父Widget管理状态并告诉其子Widget何时更新。👇示例中:TapboxB类继承StatelessWidget不管理任何状态。状态由父组件管理,因此它的父组件为StatefulWidget

Dart 复制代码
class ParentWidget extends StatefulWidget {
  @override
  _ParentWidgetState createState() => _ParentWidgetState();
}
class _ParentWidgetState extends State<ParentWidget> {
  bool _active = false;//管理_active状态
  void _handleTapboxChanged(bool newValue) {
    setState(() {_active = newValue;});//当状态改变时,调用setState()更新UI。
  }
  @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);}
  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[700] : Colors.grey[600],),),
    );
  }
}

混合状态管理

对于一些组件来说,自身管理一些外观状态,而父组件管理一些其他外部状态。

在例子中,增加一个子widget点击是有高亮效果状态_highlight。当按下、抬起、或者取消点击时更新_highlight状态。

Dart 复制代码
class ParentWidgetC extends StatefulWidget {
  @override
  _ParentWidgetCState createState() => _ParentWidgetCState();
}
class _ParentWidgetCState extends State<ParentWidgetC> {
  bool _active = false;//点击盒子_active状态改变,调用setState()更新UI。
  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;
  @override
  _TapboxCState createState() => _TapboxCState();
}
class _TapboxCState extends State<TapboxC> {
  bool _highlight = false;//管理_highlight 状态
  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[700] : Colors.grey[600],
          border: _highlight? Border.all(color: Colors.teal[700],width: 10.0,): null,),),);
  }
}

在这个案例中,我们复习了一个知识点:

Flutter Widget:StatefulWidget&StatelessWidget&State

一个 StatefulWidget 类可以有一个或多个State实例,而State实例只关联一个widget 实例。


全局状态管理

当应用中需要一些跨组件(包括跨路由)的状态需要同步时,上面介绍的方法便很难胜任了。

比如,我们有一个设置页,里面可以设置应用的语言,我们为了让设置实时生效,我们期望在语言状态发生改变时,App中依赖应用语言的组件能够重新 build 一下,但这些依赖应用语言的组件和设置页并不在一起,所以这种情况用上面的方法很难管理。

这时,正确的做法是通过一个全局状态管理器来处理这种相距较远的组件之间的通信。

目前主要有两种办法:

  1. 实现一个全局的事件总线。将语言状态改变对应为一个事件,然后在APP中依赖应用语言的组件的initState 方法中订阅语言改变的事件。当用户在设置页切换语言后,我们发布语言改变事件,而订阅了此事件的组件就会收到通知,收到通知后调用setState(...)方法重新build一下自身即可。
  2. 使用一些专门用于状态管理的包,如 Provider、Redux。
相关推荐
qiyi.sky2 分钟前
JavaWeb——Vue组件库Element(3/6):常见组件:Dialog对话框、Form表单(介绍、使用、实际效果)
前端·javascript·vue.js
煸橙干儿~~6 分钟前
分析JS Crash(进程崩溃)
java·前端·javascript
安冬的码畜日常15 分钟前
【D3.js in Action 3 精译_027】3.4 让 D3 数据适应屏幕(下)—— D3 分段比例尺的用法
前端·javascript·信息可视化·数据可视化·d3.js·d3比例尺·分段比例尺
l1x1n042 分钟前
No.3 笔记 | Web安全基础:Web1.0 - 3.0 发展史
前端·http·html
昨天;明天。今天。1 小时前
案例-任务清单
前端·javascript·css
zqx_72 小时前
随记 前端框架React的初步认识
前端·react.js·前端框架
惜.己2 小时前
javaScript基础(8个案例+代码+效果图)
开发语言·前端·javascript·vscode·css3·html5
什么鬼昵称3 小时前
Pikachu-csrf-CSRF(get)
前端·csrf
长天一色3 小时前
【ECMAScript 从入门到进阶教程】第三部分:高级主题(高级函数与范式,元编程,正则表达式,性能优化)
服务器·开发语言·前端·javascript·性能优化·ecmascript
NiNg_1_2343 小时前
npm、yarn、pnpm之间的区别
前端·npm·node.js