Flutter 中的 MVVM 架构实现指南

本文首发于公众号:移动开发那些事:Flutter 中的 MVVM 架构实现指南

1 MVVM架构

MVVM (Model-View-ViewModel) 是一种软件架构模式,它是 MVC(Model-View-Controller)MVP(Model-View-Presenter)的进化版本, 它将应用程序的用户界面(UI)逻辑与业务逻辑分离,通过 ViewModel 作为中间层连接 ViewModel

1.1 核心组件

  • Model:负责数据处理、业务逻辑和网络请求等,不直接与UI交互;
  • ViewUI 界面,负责展示数据和接收用户输入,通过与ViewModel的绑定来进行UI的更新;
  • ViewModel:连接 ViewModel 的桥梁,处理视图逻辑并提供数据给View

1.2 架构特点

  • 双向数据绑定ViewViewModel 之间的数据变化会自动同步,减少了手动更新 UI 的代码;
  • 提高可测试性ViewModel 可以独立测试,无需依赖UI
  • 分离关注点:UI 逻辑和业务逻辑分离,提高代码可维护性和可测试性。
  • 提高可维护性:代码的可读性和可维护性更高,适合大型复杂项目的开发;

1.3 适用场景

  • 复杂 UI 应用:当界面逻辑复杂时,MVVM 能有效管理状态和数据流;
  • 需要频繁更新 UI 的应用:双向数据绑定使数据更新更高效;
  • 需要单元测试的项目ViewModel 的独立性使其易于测试;

2 简单架构示例

以一个简单的计数器的例子来说明如何来搭建一个MVVM的架构;

php 复制代码
import 'package:flutter/material.dart';

// 步骤1:定义 Model: 负责数据处理
class CounterModel {
  int _count = 0;
  
  int get count => _count;
  
  void increment() {
    _count++;
  }
  
  void decrement() {
    _count--;
  }
}

// 步骤2:ViewModel: 连接 View 和 Model
class CounterViewModel with ChangeNotifier {
  final CounterModel _model = CounterModel();
  
  int get count => _model.count;
  
  void increment() {
  	// 处理数据更新
    _model.increment();
    notifyListeners(); // 通知 View 更新
  }
  
  void decrement() {
  	// 处理数据更新
    _model.decrement();
    notifyListeners(); // 通知 View 更新
  }
}


// 步骤3 View: UI 界面
class CounterScreen extends StatelessWidget {
  final CounterViewModel _viewModel = CounterViewModel();
  
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: Text('MVVM 计数器')),
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: [
            // 使用 ValueListenableBuilder 监听 ViewModel 变化,viewModel变化时,对应的页面也会同步刷新
            ValueListenableBuilder(
              valueListenable: _viewModel,
              builder: (context, value, child) {
                return Text(
                  '计数: ${_viewModel.count}',
                  style: TextStyle(fontSize: 24),
                );
              },
            ),
            SizedBox(height: 20),
            Row(
              mainAxisAlignment: MainAxisAlignment.center,
              children: [
                ElevatedButton(
                  onPressed: _viewModel.decrement,
                  child: Text('减'),
                ),
                SizedBox(width: 20),
                ElevatedButton(
                  onPressed: _viewModel.increment,
                  child: Text('加'),
                ),
              ],
            ),
          ],
        ),
      ),
    );
  }
}

3 使用GetX框架实现MVVM架构

GetX 是 Flutter 中一个强大的轻量级框架,它简化了状态管理、路由和依赖注入等功能。下面是使用 GetX 实现相同计数器应用的示例:

php 复制代码
import 'package:flutter/material.dart';
import 'package:get/get.dart';

// 步骤1:定义 Model: 负责数据处理
class CounterModel {
  int _count = 0;
  
  int get count => _count;
  
  void increment() {
    _count++;
  }
  
  void decrement() {
    _count--;
  }
}

// 步骤2:ViewModel (在 GetX 中称为 Controller)
class CounterController extends GetxController {
  final CounterModel _model = CounterModel();
  
  // 使用 RxInt 实现响应式数据
  final RxInt _count = 0.obs;
  
  int get count => _count.value;
  
  void increment() {
    _model.increment();
    _count.value = _model.count; // 更新响应式数据
  }
  
  void decrement() {
    _model.decrement();
    _count.value = _model.count; // 更新响应式数据
  }
}

// 步骤3 : View: UI 界面
class CounterScreen extends StatelessWidget {
  // 使用 Get.put() 初始化 Controller
  final CounterController _controller = Get.put(CounterController());
  
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: Text('GetX MVVM 计数器')),
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: [
            // 使用 Obx 监听 Controller 中的响应式数据
            Obx(() => Text(
                  '计数: ${_controller.count}',
                  style: TextStyle(fontSize: 24),
                )),
            SizedBox(height: 20),
            Row(
              mainAxisAlignment: MainAxisAlignment.center,
              children: [
                ElevatedButton(
                  onPressed: _controller.decrement,
                  child: Text('减'),
                ),
                SizedBox(width: 20),
                ElevatedButton(
                  onPressed: _controller.increment,
                  child: Text('加'),
                ),
              ],
            ),
          ],
        ),
      ),
    );
  }
}

4 适用场景

场景类型 推荐方案 原因
小型应用/快速原型 GetX 开发速度快,代码少
中大型项目 原生 + Provider 分层明确,长期维护性强
多平台代码共享 原生 + 抽象ViewModel 业务逻辑与平台解耦
高频更新 GetX 性能优化,避免全页重建

5 总结

MVVM 架构模式在 Flutter 开发中非常有用,它可以帮助我们构建更清晰、更易维护的应用程序。在不使用第三方框架的情况下,我们可以通过ChangeNotifierStream 等机制实现 MVVM,但代码会相对复杂。而使用 GetX 这样的框架可以大大简化 MVVM 的实现,提供更简洁的代码和更好的开发体验。实际使用过程中,可以根据业务的复杂程度,团队熟练度,以及长期计划来决定使用哪个机制来实现MVVM架构;

6 参考文献

相关推荐
前端互助会23 分钟前
Live2D形象展示与文本语音播报:打造生动交互体验的完整实现
前端·vue.js·microsoft·交互
努力的小郑2 小时前
今晚Cloudflare一哆嗦,我的加班计划全泡汤
前端·后端·程序员
q***64973 小时前
头歌答案--爬虫实战
java·前端·爬虫
凌波粒3 小时前
SpringMVC基础教程(4)--Ajax/拦截器/文件上传和下载
java·前端·spring·ajax
液态不合群4 小时前
DDD驱动低代码开发:从业务流程到领域模型的全链路设计
前端·低代码·架构·ddd
jonyleek4 小时前
JVS低代码开发中,如何创建自定义前端页面并接入到现有系统中,从创建到接入的全攻略
前端·低代码·前端框架·软件开发
谢尔登4 小时前
【React】React组件的渲染过程分为哪几个阶段?
前端·javascript·react.js
MediaTea4 小时前
Python 第三方库:Flask(轻量级 Web 框架)
开发语言·前端·后端·python·flask
5***o5004 小时前
前端构建工具缓存清理,解决依赖问题
前端·缓存
lcc1875 小时前
Vue Vue与VueComponent的关系
前端·vue.js