dva 首先是一个基于 redux 和 redux-saga 的数据流方案,然后为了简化开发体验,dva 还额外内置了 react-router 和 fetch。
Dva 概念
dva = React-Router + Redux + Redux-saga
1 | import dva from 'dva'; |
- State:一个对象,保存整个应用状态
- View:React 组件构成的视图层
- Action:一个对象,描述事件
- connect 方法:一个函数,绑定 State 到 View
- dispatch 方法:一个函数,发送 Action 到 State
改变数据的时候可以通过 dispatch 发起一个 action,
同步行为:直接通过 Reducers 改变 State ,异步行为:会先触发 Effects 然后流向 Reducers 最终改变 State.
Models
State : 是储存数据的地方,收到 Action 以后,会更新数据。
type State = any
表示 Model 的状态数据
Action
type AsyncAction = any
Action 是一个普通 javascript 对象,它是改变 State 的唯一途径。无论是从 UI 事件、网络回调,还是 WebSocket 等数据源所获得的数据,最终都会通过 dispatch 函数调用一个 action,从而改变对应的数据。action 必须带有 type 属性指明具体的行为,其它字段可以自定义,如果要发起一个 action 需要使用 dispatch 函数;需要注意的是 dispatch 是在组件 connect Models以后,通过 props 传入的。
1 | dispatch({ |
connect:绑定 State 到 View
1 | import { connect } from 'dva'; |
mapStateToProps 函数会返回一个对象,用于建立 State 到 Props 的映射关系。
dispatch 函数:用来将 Action 发送给 State。
type dispatch = (a: Action) => Action
dispatching function 是一个用于触发 action 的函数,被 connect 的 Component 会自动在 props 中拥有 dispatch 方法。
1 | dispatch({ |
Reducer
type Reducer<S, A> = (state: S, action: A) => S
Reducer(也称为 reducing function)函数接受两个参数:之前已经累积运算的结果和当前要被累积的值,返回的是一个新的累积结果。该函数把一个集合归并成一个单值。
Effect
Effect 被称为副作用,在我们的应用中,最常见的就是异步操作。它来自于函数编程的概念,之所以叫副作用是因为它使得我们的函数变得不纯,同样的输入不一定获得同样的输出。计算以外的操作都属于 Effect,典型的就是 I/O 操作、数据库读写。Effect 是一个 Generator 函数,内部使用 yield 关键字,标识每一步的操作(不管是异步或同步)。
dva 提供多个 effect 函数内部的处理函数,比较常用的是 call 和 put。
- call:执行异步函数
- put:发出一个 Action,类似于 dispatch
dva 为了控制副作用的操作,底层引入了redux-sagas做异步流程控制,由于采用了generator的相关概念,所以将异步转成同步写法,从而将effects转为纯函数。至于为什么我们这么纠结于 纯函数,如果你想了解更多可以阅读Mostly adequate guide to FP,或者它的中文译本JS函数式编程指南。
Subscription
Subscriptions 是一种从 源 获取数据的方法,它来自于 elm。
Subscription 语义是订阅,用于订阅一个数据源,然后根据条件 dispatch 需要的 action。数据源可以是当前的时间、服务器的 websocket 连接、keyboard 输入、geolocation 变化、history 路由变化等等。
Router
这里的路由通常指的是前端路由,由于我们的应用现在通常是单页应用,所以需要前端代码来控制路由逻辑,通过浏览器提供的 History API 可以监听浏览器url的变化,从而控制路由相关操作。
dva 实例提供了 router 方法来控制路由,使用的是react-router。
通信:组件之间如何通信?
- 向子组件发消息
- 向父组件发消息
- 向其他组件发消息
数据流:数据如何和视图串联起来?路由和数据如何绑定?如何编写异步逻辑?等等
目前流行的数据流方案有:
- Flux,单向数据流方案,以 Redux 为代表
- Reactive,响应式数据流方案,以 Mobx 为代表
- 其他,比如 rxjs 等
- 最流行的社区 React 应用架构方案如下。
路由: React-Router
架构: Redux
异步操作: Redux-saga