前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >推荐一种简单的在Flutter中分离View与Model的方法

推荐一种简单的在Flutter中分离View与Model的方法

作者头像
HowHardCanItBe
发布于 2020-09-15 07:40:04
发布于 2020-09-15 07:40:04
1.5K00
代码可运行
举报
文章被收录于专栏:从Android到Flutter从Android到Flutter
运行总次数:0
代码可运行

问题

我们在做Flutter开发的时候主要会在State中加入很多自己的业务逻辑,例如网络请求,数据处理等等,如果你的业务逻辑比较复杂的话会面对着一个越来越膨胀的State。代码的可读性下降,日后维护也越来越困难。这和我们在开发Android的时候遇到巨无霸Activity是同样的问题。解决办法就是分层解耦。Android从MVC进化到MVP/MVVM。Flutter 也有开发者把MVP引入到Flutter来解决这个问题。这里我们来看另一种比较简单的方法。

方法

我们先来看一下官方的那个原始的Counter例子:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
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还不太了解的同学可以找相关的文章看一下。改造以后的代码如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
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里。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
mixin _CounterStateMixin < T extends StatefulWidget> on State<T> {
  int _counter = 0;
  void _incrementCounter() {
    setState(() {
      _counter++;
    });
  }

}

然后原来的State只需要混入这个mixin就好了。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class _CounterState extends State<CounterPage> with _CounterStateMixin

这里我们就把View和Model分开了,View相关的逻辑都在State中,而Model相关的逻辑则都在StateMixin里。

是不是很简单?如果用MVP或者其他方式来实现解耦的话很可能需要多创建几个类,写很多模板代码,引入第三方库,甚至需要IDE插件的帮助。

另外一个优点就是副作用小,我们都知道使用mixin的话在运行时可以认为完全和原来那个State是一致的。如果使用MVP的话你可能需要自己处理State的生命周期,否则有可能会遇到内存泄漏或者空指针等问题。

另外,这种方式也可以配合Provider等其他状态管理机制运行,可以说十分友好了。

完整代码如下,大家感兴趣可以试着跑一下试试:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
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就可以了:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class _CounterPageState extends State<CounterPage> with _CounterStateMixin 

class _NewCounterPage1State extends State<NewCounterPage> with _CounterStateMixin 

关于生命周期,由于这个mixin是对State的扩展,所以与生命周期相关的函数如initState()didUpdateWidget()dispose()等都可以在mixin中覆写,例如说网络请求就可以放在StateMixininitState()函数里。

总之,我们的目的是View与Model分离,所以要尽可能的把与视图相关的逻辑放在State中,例如构建Widget树相关的逻辑,动画相关的逻辑等。而与Model相关的逻辑则尽量放在StateMixin里,例如网络请求等。

以上就是对使用mixin来实现Flutter中View与Model分离的介绍,大家看完如果有什么想法欢迎评论。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Git常用命令总结
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
拓荒者
2019/09/25
5740
Git 常用命令
Git 的工作就是创建和保存你项目的快照及与之后的快照进行对比。常用的是以下 6 个命令:
Yorkyu
2022/03/22
5160
Git 常用命令
Git常用命令和常见问题
你是 Windows 程序员,且正在开发仅运行在 Windows 上的项目,可以设置 false 取消此功能,把回车保留在版本库中:
青梅煮码
2023/02/18
6370
Git常用命令和常见问题
git的一些常用命令
git init 把该目录变成Git可以管理的仓库 git add file 将文件添加入暂存区 git commit -m "提交描述" 提交到仓库
meihuasheng
2021/03/18
4060
Git 常用命令总结
如果git pull提示“no tracking information”,则说明本地分支和远程分支的链接关系没有创建,用命令git branch –set-upstream branch-name origin/branch-name。
颍川
2019/11/20
4100
Git常用命令速查表
Git的设置文件为.gitconfig,它可以在用户主目录下(全局配置),也可以在项目目录下(项目配置)。
Rookie
2018/08/13
4030
01 . Git常用命令及方法和分支管理
Git同样告诉我们,用命令git reset HEAD file可以把暂存区的修改撤销掉(unstage),重新放回工作区:
iginkgo18
2020/09/27
5480
01 . Git常用命令及方法和分支管理
Git技术干货!工作中Git的使用实践和常用命令合集!
从本质上来讲 Git 是一个内容寻址(content-addressable)文件系统,并在此之上提供了一个版本控制系统的用户界面。
洋仔聊编程
2020/08/02
5520
Git 常用命令总结,掌握这些,轻松驾驭版本管理
最近公司的代码管理工具要从SVN转到Git上,因此虽然之前用过Git,但是都是一些简单的推送提交,因此还是有必要进行一些系统的学习,这里做一下笔记,以备后询,且不定期更新。
ConardLi
2019/10/31
4830
Git 常用命令总结,掌握这些,轻松驾驭版本管理
Git常用命令
Git 是一个很强大的分布式版本控制系统。它不但适用于管理大型开源软件的源代码,管理私人的文档和源代码也有很多优势。 Git常用操作命令: 1) 远程仓库相关命令 检出仓库:$ git clone git://github.com/jQuery/jquery.git 查看远程仓库:$ git remote -v 添加远程仓库:$ git remote add [name] [url] 删除远程仓库:$ git remote rm [name] 修改远程仓库:$ git remote set-url --pu
Java帮帮
2018/03/16
1.1K0
最全的Git常用命令速查-2021版
虽然如今已有很多可视化的 Git 工具,但是很多时候我们还是需要用到命令直接操作。
小锟哥哥
2022/05/10
3180
最全的Git常用命令速查-2021版
Git常用命令总结
平常我们开发就是拷贝远程仓库中的一个分支,基于该分支进行开发。在开发过程中就是对工作区的操作。
Anymarvel
2018/10/22
5.2K0
Git常用命令总结
相关推荐
Git常用命令总结
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档