热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

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

问题我们在做Flutter开发的时候主要会在State中加入很多自己的业务逻辑,例如网络请求,数据处理等等,如果你的业务逻辑比较复杂的话

问题

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

方法

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

class _MyHomePageState extends State {int _counter = 0;void _incrementCounter() {setState(() {_counter++;});}@overrideWidget build(BuildContext context) {return Scaffold(appBar: AppBar(title: Text(widget.title),),body: Center(child: Column(mainAxisAlignment: MainAxisAlignment.center,children: [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 on State {int _counter = 0;void _incrementCounter() {setState(() {_counter++;});}}class _CounterState extends State with _CounterStateMixin {@overrideWidget build(BuildContext context) {return Scaffold(appBar: AppBar(title: Text(widget.title),),body: Center(child: Column(mainAxisAlignment: MainAxisAlignment.center,children: [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 on State {int _counter = 0;void _incrementCounter() {setState(() {_counter++;});}}

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

class _CounterState extends State 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 {@overrideWidget 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 with _CounterStateMixin {@overrideWidget build(BuildContext context) {return Scaffold(appBar: AppBar(title: Text(widget.title),),body: Center(child: Column(mainAxisAlignment: MainAxisAlignment.center,children: [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 on State {int _counter = 0;void _incrementCounter() {setState(() {_counter++;});}}

还有一点就是这个拆出来的StateMixin是可以复用的,例如你想在页面上放两个功能相同但是显示不一样的counter,让两个counter的State都混入同一个CounterStateMixin就可以了:

class _CounterPageState extends State with _CounterStateMixin class _NewCounterPage1State extends State with _CounterStateMixin

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

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

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


作者:ad6623
链接:https://juejin.im/post/6844904168746926087
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。


推荐阅读
  • 深入解析 Android IPC 中的 Messenger 机制
    本文详细介绍了 Android 中基于消息传递的进程间通信(IPC)机制——Messenger。通过实例和源码分析,帮助开发者更好地理解和使用这一高效的通信工具。 ... [详细]
  • ListView简单使用
    先上效果:主要实现了Listview的绑定和点击事件。项目资源结构如下:先创建一个动物类,用来装载数据:Animal类如下:packagecom.example.simplelis ... [详细]
  • 本文介绍了在Android项目中实现时间轴效果的方法,通过自定义ListView的Item布局和适配器逻辑,实现了动态显示和隐藏时间标签的功能。文中详细描述了布局文件、适配器代码以及时间格式化工具类的具体实现。 ... [详细]
  • Android开发技巧:实现带描边的圆角图片
    本文介绍了一种在Android应用中实现带描边的圆角图片的方法。通过使用BitmapShader类,开发者可以轻松地为图片添加圆角和描边效果,提升应用的视觉体验。 ... [详细]
  • 纵向|发生_ListView和EditText使用解决方案 ... [详细]
  • 本文介绍如何通过自定义控件LoadLayout实现ListView的上拉加载更多和下拉刷新功能。LoadLayout支持上拉加载,而下拉刷新则利用了android.support.v4.widget.SwipeRefreshLayout组件。 ... [详细]
  • springMVC JRS303验证 ... [详细]
  • 本文介绍如何利用QFileSystemModel进行目录的浏览、创建及删除操作,并提供了一个简单的对话框界面实现。 ... [详细]
  • Django Admin 插件详解与应用
    本文介绍了 Django Admin 的主要功能及其在项目开发中的作用,包括如何通过模型类操作数据库、自定义 Admin 方法以及多种配置选项,旨在帮助开发者快速掌握 Django Admin 的使用技巧。 ... [详细]
  • Flutter入门指南:实现自动关闭的对话框与提示
    本文为Flutter系列教程的一部分,专注于讲解如何在Flutter应用中实现自动关闭的对话框和提示。通过具体的代码示例,帮助开发者掌握SnackBar、BottomSheet和Dialog的使用方法。 ... [详细]
  • Flutter 高德地图插件使用指南
    本文档详细介绍了如何在Flutter项目中集成和使用高德地图插件,包括安装、配置及基本使用方法。 ... [详细]
  • 本文详细介绍了如何在Android 4.4及以上版本中配置WebView以实现内容的自动高度调整和屏幕适配,确保中文显示正常,并提供代码示例。 ... [详细]
  • 本文详细介绍了 Android 开发中 layout_gravity 属性的使用方法及其在不同布局下的效果,旨在帮助开发者更好地理解和利用这一属性来精确控制视图的布局。 ... [详细]
  • 深入理解设计模式之观察者模式
    本文详细介绍了观察者模式,这是一种行为设计模式,适用于当对象状态发生变化时,需要通知其他相关对象的场景。文中不仅解释了观察者模式的基本概念,还通过Java代码示例展示了其实现方法。 ... [详细]
  • 本文探讨了Java编程中MVC模式的优势与局限,以及如何利用Java开发一款基于鸟瞰视角的赛车游戏。 ... [详细]
author-avatar
徐筱毅信贤季善
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有