推荐一种简单的在Flutter中分离View与Model的方法
问题
我们在做Flutter开发的时候主要会在State
中加入很多自己的业务逻辑,例如网络请求,数据处理等等,如果你的业务逻辑比较复杂的话会面对着一个越来越膨胀的State
。代码的可读性下降,日后维护也越来越困难。这和我们在开发Android的时候遇到巨无霸Activity
是同样的问题。解决办法就是分层解耦。Android从MVC进化到MVP/MVVM。Flutter 也有开发者把MVP引入到Flutter来解决这个问题。这里我们来看另一种比较简单的方法。
方法
我们先来看一下官方的那个原始的Counter例子:
class _MyHomePageState extends State<MyHomePage> {
int _counter = 0;
void _incrementCounter() {
setState(() {
_counter++;
});
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(widget.title),
),
body: Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
Text(
'You have pushed the button this many times:',
),
Text(
'$_counter',
style: Theme.of(context).textTheme.display1,
),
],
),
),
floatingActionButton: FloatingActionButton(
onPressed: _incrementCounter,
tooltip: 'Increment',
child: Icon(Icons.add),
),
);
}
}
可以看到,在这个_MyHomePageState
类中,视图相关的代码都在build()
这个函数体内,数据属性_counter
以及相关的函数_incrementCounter()
都存在于同一个类中。可以想象一下,如果你的页面比较复杂的话有可能会把部分视图相关的代码从build()
中拆分出来放入类似getMyWidget()
的函数,View与Model混合在一起,这个State
将会变得难以维护。
为了将View与Model分离,我们采取mixin
这种办法。对mixin
还不太了解的同学可以找相关的文章看一下。改造以后的代码如下:
mixin _CounterStateMixin < T extends StatefulWidget> on State<T> {
int _counter = 0;
void _incrementCounter() {
setState(() {
_counter++;
});
}
}
class _CounterState extends State<CounterPage> with _CounterStateMixin {
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(widget.title),
),
body: Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
Text(
'Mixin, You have pushed the button this many times:',
),
Text(
'$_counter',
style: Theme.of(context).textTheme.display1,
),
],
),
),
floatingActionButton: FloatingActionButton(
onPressed: _incrementCounter,
tooltip: 'Increment',
child: Icon(Icons.add),
),
);
}
}
首先新建一个mixin
,这里命名为_CounterStateMixin
,把原来State
中的_counter
和_incrementCounter()
挪到这个新的mixin
里。
mixin _CounterStateMixin < T extends StatefulWidget> on State<T> {
int _counter = 0;
void _incrementCounter() {
setState(() {
_counter++;
});
}
}
然后原来的State
只需要混入这个mixin
就好了。
class _CounterState extends State<CounterPage> with _CounterStateMixin
这里我们就把View和Model分开了,View相关的逻辑都在State
中,而Model相关的逻辑则都在StateMixin
里。
是不是很简单?如果用MVP或者其他方式来实现解耦的话很可能需要多创建几个类,写很多模板代码,引入第三方库,甚至需要IDE插件的帮助。
另外一个优点就是副作用小,我们都知道使用mixin
的话在运行时可以认为完全和原来那个State
是一致的。如果使用MVP的话你可能需要自己处理State
的生命周期,否则有可能会遇到内存泄漏或者空指针等问题。
另外,这种方式也可以配合Provider等其他状态管理机制运行,可以说十分友好了。
完整代码如下,大家感兴趣可以试着跑一下试试:
import 'package:flutter/material.dart';
void main() {
runApp(MyApp());
}
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
home: CounterPage(title: 'Flutter Demo Home Page'),
);
}
}
class CounterPage extends StatefulWidget {
CounterPage({Key key, this.title}) : super(key: key);
final String title;
@override
_CounterState createState() => _CounterState();
}
class _CounterState extends State<CounterPage> with _CounterStateMixin {
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(widget.title),
),
body: Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
Text(
'Mixin, You have pushed the button this many times:',
),
Text(
'$_counter',
style: Theme.of(context).textTheme.display1,
),
],
),
),
floatingActionButton: FloatingActionButton(
onPressed: _incrementCounter,
tooltip: 'Increment',
child: Icon(Icons.add),
),
);
}
}
mixin _CounterStateMixin < T extends StatefulWidget> on State<T> {
int _counter = 0;
void _incrementCounter() {
setState(() {
_counter++;
});
}
}
还有一点就是这个拆出来的StateMixin
是可以复用的,例如你想在页面上放两个功能相同但是显示不一样的counter,让两个counter的State
都混入同一个CounterStateMixin
就可以了:
class _CounterPageState extends State<CounterPage> with _CounterStateMixin
class _NewCounterPage1State extends State<NewCounterPage> with _CounterStateMixin
关于生命周期,由于这个mixin
是对State
的扩展,所以与生命周期相关的函数如initState()
,didUpdateWidget()
,dispose()
等都可以在mixin
中覆写,例如说网络请求就可以放在StateMixin
的initState()
函数里。
总之,我们的目的是View与Model分离,所以要尽可能的把与视图相关的逻辑放在State
中,例如构建Widget树相关的逻辑,动画相关的逻辑等。而与Model相关的逻辑则尽量放在StateMixin
里,例如网络请求等。
以上就是对使用mixin
来实现Flutter中View与Model分离的介绍,大家看完如果有什么想法欢迎评论。
- 优化算法——模拟退火算法
- 绘制动态心形图案::R语言绘制心形图
- 物化视图中的统计信息导致的查询问题分析和修复 (r7笔记第47天)
- R语言之系统聚类(层次)分析之图谱形式完整版
- Java操作数据库Spring(1)
- python基础知识——内置数据结构(集合)
- 关于db link权限分配的苦旅(二)(r7笔记第45天)
- 简单易学的机器学习算法——在线顺序极限学习机OS-ELM
- Java操作数据库Spring(2)
- 解决SSH连接linux中文显示乱码问题
- 设计模式——类图以及类与类之间的关系
- sysdba登录报错insufficient privileges的原因分析(r7笔记第64天)
- JDBC基础入门(1)
- LeetCode——Add Two Numbers
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- LeetCode53|搜索二维矩阵II
- LeetCode54|二叉树的层次遍历
- LeetCode55|二叉树的层次遍历II
- LeetCode56|二叉树的层平均值
- LeetCode57|二叉树的锯齿形层次遍历
- LeetCode58|N叉树的层序遍历
- 技术创作101训练营-CRC校验没那么难
- 栈与队列:有没有想过计算机是如何处理表达式的?
- 栈与队列:滑动窗口里求最大值引出一个重要数据结构
- 栈与队列:求前 K 个高频元素和队列有啥关系?
- 手写一个抖音视频去水印工具,千万别刚一个程序员
- ApplicationListener接口实践
- 浅谈vue+element全局loading加载
- LeetCode59|重复N次的元素
- 谈Vue组件的is特性