当前位置: 首页 > news >正文

Flutter MobX 响应式原理与实战详解

📚 Flutter 状态管理系列文章目录
  1. setState() 使用详解:原理及注意事项

  2. Flutter 状态管理(setState、InheritedWidget、 Provider 、Riverpod、 BLoC / Cubit、 GetX 、MobX 、Redux)

  3. InheritedWidget 组件使用及原理

  4. Flutter 中 Provider 的使用、注意事项与原理解析(含代码实战)

  5. GetX 用法详细解析以及注意事项

  6. Flutter BLoC 使用详细解析

  7. Flutter MobX 响应式原理与实战详解

✨ 起点:为什么要了解 MobX?

作为一个刚开始接触 Flutter 状态管理的开发者,我最初接触到的是 setStateProvider 等方式。当我看到有人说 MobX 是“最像魔法的状态管理库”时,我开始好奇:为什么它能不写 setState 就自动刷新 UI?

带着这样的疑问,我开启了对 MobX 的深入探索。这篇文章就是我把所有探索过程系统整理后的总结,希望对你也有帮助。


🛠 一、MobX 的安装与基本使用

MobX 的核心有三大类:

概念说明
Observable可观察状态,类似变量。变化时自动触发 UI 更新
Action改变状态的行为,所有状态变更建议通过 action
Reaction监听变化后做副作用,比如打印日志、导航等

1. 添加依赖

pubspec.yaml 中:

dependencies:flutter:sdk: fluttermobx: ^2.2.0flutter_mobx: ^2.0.6dev_dependencies:build_runner: ^2.4.6mobx_codegen: ^2.4.0

2.创建一个 MobX Store(状态容器)

counter_store.dart

import 'package:mobx/mobx.dart';part 'counter_store.g.dart'; // 自动生成的文件class CounterStore = _CounterStore with _$CounterStore;abstract class _CounterStore with Store {int count = 0;void increment() {count++;}
}

生成代码命令:

flutter pub run build_runner build
# 或者自动监听:
flutter pub run build_runner watch

3. 使用 Observer 监听状态

import 'package:flutter/material.dart';
import 'package:flutter_mobx/flutter_mobx.dart';
import 'counter_store.dart';class CounterPage extends StatelessWidget {final CounterStore counter = CounterStore();Widget build(BuildContext context) {return Scaffold(appBar: AppBar(title: Text("MobX Counter")),body: Center(child: Observer(builder: (_) => Text('${counter.count}',style: TextStyle(fontSize: 40),),),),floatingActionButton: FloatingActionButton(onPressed: counter.increment,child: Icon(Icons.add),),);}
}

📚 二、MobX 核心注解说明

注解作用示例
@observable可监听变量@observable int count = 0;
@action状态改变函数@action void increment() => count++;
@computed派生状态String get fullName => "$firstName $lastName";

🧩 三、不同类型的 Observer 使用方式

类型使用场景是否支持 child是否自动跟踪依赖
Observer通用 UI 刷新
Observer.builder自定义构建器
ObserverWidget(自定义)可封装为组件
Observer + child:子组件不依赖状态,可优化性能

1️⃣ 最常用的 Observer

适用于所有直接在 build() 方法中监听 observable 状态的场景。

✅ 示例:
Observer(builder: (_) => Text('${counter.count}',style: TextStyle(fontSize: 32),),
)

每当 counter.count 更新时,Text 会自动重建。


2️⃣ Observer + child 优化重建

用于优化性能:当 UI 中部分元素是静态的,可以通过 child 参数避免重建。

✅ 示例:
Observer(child: Icon(Icons.favorite), // 不会因为状态变化而重建builder: (_, child) => Row(mainAxisAlignment: MainAxisAlignment.center,children: [child!,SizedBox(width: 8),Text('${counter.count}'),],),
)

Icon 是静态内容,MobX 不会重复创建,节省性能。


3️⃣ 封装成组件:自定义 ObserverWidget

当你想将 Observer 封装成复用组件时,可创建一个继承自 StatelessWidgetObserver 控件。

✅ 示例:
class CountText extends StatelessWidget {final CounterStore store;const CountText({required this.store});Widget build(BuildContext context) {return Observer(builder: (_) => Text('${store.count}', style: TextStyle(fontSize: 24)),);}
}

使用方式:

CountText(store: counterStore)

4️⃣ ListView/GridView 中的 Observer

当你需要对列表内容做响应式刷新时,每个 item 可以使用 Observer 包裹。

✅ 示例:
Observer(builder: (_) => ListView.builder(itemCount: todoStore.todos.length,itemBuilder: (_, index) {final todo = todoStore.todos[index];return Observer(builder: (_) => CheckboxListTile(value: todo.done,onChanged: (_) => todo.toggleDone(),title: Text(todo.title),),);},),
)

每个 CheckboxListTile 都会监听自己对应的 todo.done,提高效率。


🧠 补充:多层 Observer 嵌套性能说明

MobX 的响应式追踪是“精确依赖感知”的,每个 Observer 只监听自己用到的 @observable不会因为 store 中其他字段变化而重建,比传统 Provider 更细粒度。


✅ 小结对比
类型优势适用场景
Observer最常用,自动追踪通用响应式 UI
Observer + child性能优化静态元素嵌入 UI
自定义 StatelessWidget 包裹 Observer组件复用通用响应式组件
列表中嵌套 Observer精细追踪每一项变化todo、商品列表

✅ MobX 为什么可以“这么做”——原理解析

一、核心机制:可观察(Observable) + 自动追踪(Tracking)
  1. @observable 的变量不是普通变量,而是被 MobX 包装成了 可追踪对象(Atom)

  2. Observer 被构建时,MobX 开启一个 “追踪上下文”(Tracking Context):

    • Observer.builder() 中访问了哪些 @observable,就自动把这些变量注册为“依赖”。
  3. 一旦这些依赖发生变化,MobX 会自动通知 Observer 重新构建 UI。


二、过程举例说明

int count = 0;Observer(builder: (_) => Text('$count'),
)

➡ MobX 做了什么?

  1. 初次 build 时读取了 count,MobX 会记录下这个“读取动作”:

    “这个 Observer 正在依赖 count!”

  2. 后续调用 count++(即修改 observable)时:

    • MobX 发现 count 改了,它就通知所有依赖 count 的观察者(也就是这个 Observer)重新执行 builder,从而刷新 UI。

三、MobX 响应式模型的类比

可以把 MobX 的响应式系统类比成:

  • @observable = 数据源(比如“温度传感器”)
  • Observer = 显示模块(比如“电子温度计”)
  • MobX = 自动布线系统,知道哪个显示器监听哪个传感器

只要传感器变化,显示模块就刷新。开发者什么都不用手动做。


四、底层依赖追踪的原理:Atom + Reaction

MobX 的核心类是:

说明
Atom每个 observable 变量都是一个 Atom
Reaction每个 Observer 内部注册了一个 reaction
Context每次执行 builder() 时,MobX 开启上下文收集依赖
Derivation指的是任何依赖 observable 的函数,比如 computedObserver

五、Flutter 中 UI 为什么可以重建?

因为 Observer 继承自 StatelessWidget,而其内部逻辑其实是这样的(伪代码):

class Observer extends StatelessWidget {final Reaction _reaction;Observer({required WidgetBuilder builder}) {_reaction = Reaction(() {// 当依赖的 observable 改变时,调用 setState 或 rebuild});}Widget build(BuildContext context) {// 启动依赖收集_reaction.track(() {builder(context); // 在这个过程中读取 observable,就会注册依赖});}
}

✅ 总结一句话

MobX 可以“这么做”,是因为它在 Observer.builder 执行时自动收集依赖变量,一旦这些变量变化,MobX 会主动通知刷新 UI。你不需要写 setState,它已经帮你自动“追踪 + 通知 + 重建”了。


⚙️ 五、自己手写一个简化版 MobX(低配实现)

太棒了!下面我就来写一个低配版的 MobX 响应式系统实现,让你直观理解它的底层机制。这个版本包括:

  1. Observable<T>:可观察变量
  2. Reaction:监听依赖并自动响应
  3. autorun():一个简化版的 Observer 行为
  4. 自动依赖追踪:MobX 的“神操作”

🧠 目标:当你读取 observable 的时候自动注册依赖,当你修改 observable 的时候,能触发依赖函数。


🔧 第一步:实现全局上下文收集机制

typedef VoidCallback = void Function();class ReactiveContext {static final ReactiveContext _instance = ReactiveContext._internal();VoidCallback? _currentReaction;factory ReactiveContext() => _instance;ReactiveContext._internal();void setCurrentReaction(VoidCallback? reaction) {_currentReaction = reaction;}VoidCallback? get currentReaction => _currentReaction;
}

🔮 第二步:实现 Observable

class Observable<T> {T _value;final Set<VoidCallback> _listeners = {};Observable(this._value);T get value {// 自动依赖收集final reaction = ReactiveContext()._currentReaction;if (reaction != null) {_listeners.add(reaction);}return _value;}set value(T newValue) {if (newValue != _value) {_value = newValue;// 通知所有监听者for (var listener in _listeners) {listener();}}}
}

🔁 第三步:实现 autorun(模拟 Observer)

void autorun(VoidCallback runner) {void wrapped() {ReactiveContext().setCurrentReaction(wrapped); // 注册当前 reactionrunner();ReactiveContext().setCurrentReaction(null); // 清理}wrapped(); // 初始执行一次,收集依赖
}

🎯 示例:模拟 Counter 自动追踪 + 响应

void main() {final count = Observable(0);autorun(() {print("当前 count 是:${count.value}");});// 模拟用户操作count.value = 1; // 自动触发 autorun 输出count.value = 2;
}
✅ 输出:
当前 count 是:0
当前 count 是:1
当前 count 是:2

🧠 总结:

MobX 组件我们实现的类
@observableObservable<T>
Observerautorun()
依赖追踪ReactiveContext
reactionVoidCallback 注册

💡 提升建议:

你可以进一步加上:

  • computed(派生值)
  • action(封装状态更新)
  • dispose()(清除监听)
  • 与 Flutter Widget 结合(如用 ValueListenableBuilder 或自写 ObserverWidget

🗺 六、MobX 响应流程图

Observable↓ (读取)
Reaction ← 自动注册依赖↓
值变化?↓ 是
刷新 UI(Observer)

(你也可以参考文中附图,完整展现了 MobX 的响应链条)


✅ 七、总结:MobX 值得用吗?

MobX 让状态管理变得优雅和现代化:

  • 不用写 setState
  • 不需要 notifyListeners
  • 自动依赖收集
  • 最细粒度响应更新

适合追求响应式编程和简洁架构的开发者。如果你喜欢 Vue、Svelte 那种“写了就动”的感觉,MobX 会让你爱不释手。


🧰 附加建议

  • 推荐将每个 Store 模块化、组件化;
  • 可结合依赖注入工具如 GetIt 使用;
  • 想要更复杂管理?可引入 reactionwhen 进行副作用管理。
📚 Flutter 状态管理系列文章目录
  1. setState() 使用详解:原理及注意事项

  2. Flutter 状态管理(setState、InheritedWidget、 Provider 、Riverpod、 BLoC / Cubit、 GetX 、MobX 、Redux)

  3. InheritedWidget 组件使用及原理

  4. Flutter 中 Provider 的使用、注意事项与原理解析(含代码实战)

  5. GetX 用法详细解析以及注意事项

  6. Flutter BLoC 使用详细解析

  7. Flutter MobX 响应式原理与实战详解

http://www.xdnf.cn/news/1068283.html

相关文章:

  • OpenLayers 上传Shapefile文件
  • docker start mysql失败,解决方案
  • 【猜你需要】使用了Git LFS还是无法上传文件?文件过大?
  • 打造丝滑的Android应用:LiveData完全教程
  • 【启发式算法】RRT*算法详细介绍(Python)
  • Oracle 数据库查询:多表查询
  • 测试平台ui自动化demo说明
  • [论文阅读] 人工智能 + 软件工程 | 探秘LLM软件代理:从黑箱决策到透明轨迹分析
  • 创客匠人 AI 赋能:创始人 IP 打造的效率革命与信任重构
  • 数的范围(连续数字边界)
  • 以太网基础②RGMII 与 GMII 转换电路设计
  • Spring Boot 系统开发:打造高效、稳定、可扩展的企业级应用
  • 学习日记-spring-day37-6.25
  • SpringCloud系列(35)--使用HystrixDashboard进行服务监控
  • OSS跨区域复制灾备方案:华东1到华南1的数据同步与故障切换演练
  • 数智时代如何构建人才培养生态?生成式人工智能(GAI)认证,引领数智时代人才培养新方向
  • Kafka如何保证消息可靠?
  • 计算机网络期末复习
  • Linux操作系统Nginx Web服务
  • JVM(12)——详解G1垃圾回收器
  • 多模态大模型(从0到1)
  • JavaEE初阶第四期:解锁多线程,从 “单车道” 到 “高速公路” 的编程升级(二)
  • 《AI大模型应用技术开发工程师》学习总结
  • 2025.6.16-实习
  • 【Linux网络与网络编程】15.DNS与ICMP协议
  • 《仿盒马》app开发技术分享-- 兑换列表展示(68)
  • AI时代工具:AIGC导航——AI工具集合
  • MySQL:深入总结锁机制
  • C++信奥赛闯关题目1
  • 有AI后,还用学编程吗?