使用MobX管理React的状态
使用 PWA 框架进行应用开发,不论是 Vue 还是 React,最关注的设计应该就是状态的管理了。React 生态下状态管理的框架可谓是五花八门,本人目前用过两个,分别是 @reduxjs/toolkit (opens new window) 和 mobx-react (opens new window)。 个人感觉 Mobx 用起来更简洁,所以本文着重介绍 Mobx。
# 安装 MobX
MobX 本来就可以独立 PWA 框架来使用,所以安装的时候需要安装本体和 React 集成库。
pnpm install mobx-react-lite mobx
1
# 设计 Store
为了测试 MobX 的使用,我写了一个简单的 Todo 例子,接下来讲解如何设计 Store 方便使用。
export class TodoList {
@observable.shallow list: TodoItem[] = [];
constructor() {
makeObservable(this);
}
@action
private fromJS = () => {
this.list = this.list.map(
todo => new TodoItem(todo.title, todo.id, todo.completed)
);
};
@action
addTodo = (text: string) => {
this.list.push(new TodoItem(text));
};
@action
removeTodo = (todo: TodoItem) => {
this.list.splice(this.list.indexOf(todo), 1);
};
@action
removeCompleted = () => {
this.list = this.activeTodos;
};
@action
toggleAll = (value: boolean) => {
this.list.forEach(todo => {
todo.updateCompleted(value);
});
};
@computed
get showTodo(): boolean {
return this.list.length > 0;
}
@computed
get allTodos(): TodoItem[] {
return this.list;
}
@computed
get completedTodos(): TodoItem[] {
return this.list.filter(todo => todo.completed);
}
@computed
get activeTodos(): TodoItem[] {
return this.list.filter(todo => !todo.completed);
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
不同于 React Hooks 函数式声明,Mobx 我们推荐使用 Class 方式来声明,通过修饰符来绑定属性和方法对应 MobX 的相关概念。MobX 推荐使用 action 进行数据的更改,使用 computed 完成对数据的筛选和处理。不建议直接更改声明的数据。
# 设计全局 Store
export type RootStore = {
todoList: TodoList;
};
export const store: RootStore = {
todoList
};
const StoreContext = React.createContext<RootStore>(store);
export const StoreProvider = StoreContext.Provider;
export const useStore = () => React.useContext(StoreContext);
1
2
3
4
5
6
7
8
9
10
11
12
13
2
3
4
5
6
7
8
9
10
11
12
13
通过声明全局 Store,模块化设计各个子 Store,这里虽然只有一个 todoList, 但可以根据应用规模添加子模块,方便扩展。
# Root 导入 Store
ReactDOM.createRoot(document.getElementById('root')!).render(
<React.StrictMode>
<StoreProvider value={store}>
<App />
</StoreProvider>
</React.StrictMode>
);
1
2
3
4
5
6
7
2
3
4
5
6
7
在 React 根节点声明 Store, 保证整个应用都可以同步到 store 的数据。
# 封装 View
我们需要在使用 store 的地方,调用 observer 保证状态的更新能及时反应到视图上。
const TodoView = () => {
const { todoList } = useStore();
return (
<section className="todoapp">
<header className="header">
<h1>todos</h1>
</header>
</section>
);
};
export default observer(TodoView);
1
2
3
4
5
6
7
8
9
10
11
12
13
2
3
4
5
6
7
8
9
10
11
12
13
# 总结
MobX 的状态更新原理是基于 Proxy (opens new window)。 通过代理声明的数据对象,在读取和写入的时候执行相关回调进行更新。