React 应用框架在蚂蚁金服的实践
陈成 (花名:云谦)
data:image/s3,"s3://crabby-images/b98f2/b98f2f6de61d2daee4c1fbe9e64eebf877fce132" alt=""
Dva
data:image/s3,"s3://crabby-images/b98f2/b98f2f6de61d2daee4c1fbe9e64eebf877fce132" alt=""
data:image/s3,"s3://crabby-images/2fb28/2fb2807e05b9558f93034746fdc3d34910d7c91e" alt=""
蚂蚁金服
data:image/s3,"s3://crabby-images/e1ef7/e1ef792b0dda7737a62cad3dd0a2a56db310478a" alt=""
data:image/s3,"s3://crabby-images/eb203/eb203aa12999e617ad3d580564fdc3d5fc7ed5f0" alt=""
杭州
data:image/s3,"s3://crabby-images/7451a/7451a3b591d15107043fbaf040b767bbae725771" alt=""
data:image/s3,"s3://crabby-images/c280e/c280ec3d00dee8e8bd9268380e8b1daa4c986904" alt=""
此次分享包含:
- 蚂蚁金服前端应用架构历程
- 介绍 Dva,基于 redux 的前端框架
- 我们的前端技术栈
data:image/s3,"s3://crabby-images/1f801/1f801c229c285fcc0ae3d3f5d684cc5ab2e55398" alt=""
Let's
Get
Started!
data:image/s3,"s3://crabby-images/ef993/ef99346d49ce97caa6cddc9e50acd1239fe574ae" alt=""
data:image/s3,"s3://crabby-images/a05ac/a05acd748d9737a7865105f80b4d7570d4612d80" alt=""
data:image/s3,"s3://crabby-images/99d57/99d578c8f7f347693deb38dd8fe0f812ee5d4468" alt=""
PlainReact
Roof
Roof@0.5
Redux
Dva
应用架构历程
Roof 0.4
- 第一个版本的应用架构
- 类似 baobab
Roof 0.5
data:image/s3,"s3://crabby-images/4f81f/4f81f2d10e108a6be51569d79c91d210d784c8e2" alt=""
Redux
data:image/s3,"s3://crabby-images/b4f87/b4f87186b537ec97b8da44e93d52cf1af41e25f2" alt=""
- 轻量级数据流方案,解决从 state 到 component 的单向数据流转问题
- 借助 Action 和 Reducer 实现可预测的 state 变更
- 社区活跃,丰富的扩展、调试方案
- ...
Redux 最佳实践
data:image/s3,"s3://crabby-images/ca8f0/ca8f026625963bc320a4998ff7cb727d13720cc7" alt=""
新的问题
- 非常多的库供选择
-
代码写哪里?是个问题
-
代码分散,影响专注力
-
一遍遍重复地写 showLoading 和 hideLoading
-
出错处理太繁琐,每个异步 saga 都要 try .. catch
-
项目太大了,我需要动态加载方案
- ...
data:image/s3,"s3://crabby-images/e0c6c/e0c6c4bcee3c0173a45a62e8f8fba62be302cbcb" alt=""
Dva
Build redux application easier and better.
data:image/s3,"s3://crabby-images/b98f2/b98f2f6de61d2daee4c1fbe9e64eebf877fce132" alt=""
data:image/s3,"s3://crabby-images/7e8e6/7e8e66e833b141051bcc5678911c21739e934295" alt=""
data:image/s3,"s3://crabby-images/670ec/670ec72921618fdc851a0e05148e1c9472ef2a1e" alt=""
data:image/s3,"s3://crabby-images/0cad0/0cad0204c8dd15540649f6351c18f5021b47a122" alt=""
Dva @ 蚂蚁金服
data:image/s3,"s3://crabby-images/b6964/b6964a2fba8ec32c8a3558f610eccfe333c8137f" alt=""
Dva 是什么
-
框架,而非类库
-
基于 redux, react-router, redux-saga 的轻量级封装
-
借鉴 elm 的概念,Reducer, Effect 和 Subscription
-
...
data:image/s3,"s3://crabby-images/b98f2/b98f2f6de61d2daee4c1fbe9e64eebf877fce132" alt=""
特性
-
仅有 5 个 API
-
支持 HMR
-
支持 SSR (ServerSideRender)
-
支持 Mobile/ReactNative
-
支持 TypeScript
-
支持路由和 Model 的动态加载
-
完善的语法分析库 dva-ast
-
...
初印象
import dva from'dva';
// 1. 创建 dva 实例
const app = dva();
// 2. 装载插件 (可选)
app.use(require('dva‐loading')());
// 3. 注册 Model
app.model(require('./models/count'));
// 4. 配置路由
app.router(require('./router'));
// 5. 启动应用
app.start('#root');
data:image/s3,"s3://crabby-images/1f801/1f801c229c285fcc0ae3d3f5d684cc5ab2e55398" alt=""
DEMO
5 个 API
-
app = dva(Opts)
-
app.use(Hooks)
-
app.model(ModelObject)
-
app.router(Function)
-
app.start([HTMLElement], opts)
8 个概念
-
State
-
Action
-
Model
-
Reducer
-
Effect
-
Subscription
-
-
Router
-
RouteComponent
-
State
-
State 表示应用的数据层,由 model 的 state 组成全局的 state
{
todos:[],
visibilityFilter:'SHOW_ALL',
}
全局 State
app.model({namespace:'todos', state: []});
app.model({namespace:'visibilityFilter', state: 'SHOW_ALL'});
Model
Action
-
Action 表示操作事件,可以是同步,也可以是异步
{
type: String,
payload: Any?,
error? Error,
}
格式
dispatch(Action);
dispatch({ type: 'todos/add', payload: 'Learn Dva' });
触发 action
Model
-
Model 非 MVC 中的 M,而是领域模型,用于把数据相关的逻辑聚合到一起
-
包含 5 个 key
-
namespace
-
state
-
reducers
-
effects
-
subscriptions
-
Model 例子
export default {
namespace: 'todos',
state: [],
reducers: {
// 保存 todos
save(state, { payload }) {
return payload;
},
// 添加 todo
add(state, { payload }) {
return [...state, payload];
},
},
effects: {
// 异步获取 todos,然后通过 action 保存
*fetch(action, { call, put }) {
const todos = yield call(service.fetchTodos);
yield put({ type: 'save', payload: todos });
},
},
}
Reducer
-
Reducer 是唯一可以修改 state 的地方,接收 state 和 action,返回新的 state 。
(state, action) => newState
格式
// good
function(state, action) {
return state + action.payload;
}
// bad
const foo = 1;
function(state, action) {
return state + action.payload + foo;
}
例子
Effect
-
Effect 用于处理异步逻辑,基于 redux-saga 实现
- 基于 Generator
*(action, effects) => void
格式
*fetch(action, { call, put, select }) {
const todos = yield call(fetchTodos);
yield put({ type: 'save', payload: todos });
},
例子
Subscription
-
Subscription 表示订阅,用于订阅一个数据源,然后按需 dispatch action。
({ history, dispatch }) => Function|void
格式
function({ dispatch, history }) {
history.listen(({ pathname }) => {
if (pathname === '/users') {
dispatch({
type: 'users/fetch',
});
}
});
},
例子
Router
-
Router 表示路由配置信息
({ history, app }) => JSXElement | Object
格式
export default function({ history }){
return(
<Router history={history}>
<Route path="/" component={App} />
</Router>
);
}
例子
RouteComponent
-
RouteComponent 表示 Router 里匹配路径的 Component,通常会绑定 model 的数据
import { connect } from 'dva';
function App() {
return <div>App</div>;
}
function mapStateToProps(state) {
return { todos: state.todos };
}
export default connect(mapStateToProps)(App);
例子
整体架构
data:image/s3,"s3://crabby-images/edb64/edb641ddc294cee6584e6fe5b7d268836973a7a4" alt=""
dva 生态
data:image/s3,"s3://crabby-images/35f3c/35f3cc15b843a3a72c688f7944053951299685fb" alt=""
蚂蚁金服前端技术栈
data:image/s3,"s3://crabby-images/8abc3/8abc3018a8fdb9d7bb32308d7bf4689bdd572070" alt=""
相关资源
data:image/s3,"s3://crabby-images/546e8/546e8e10c4ae9def1aca62d9530e69b7885d22c1" alt=""
-
awesome-dva - dva 资源列表
-
dva-knowledge - 包含使用 dva 所需的所有知识点
-
视频教程 - 一步步创建一个应用
Thank you!
data:image/s3,"s3://crabby-images/7b34b/7b34bb800f4295574fd7aac369a709cbfb4d0818" alt=""
data:image/s3,"s3://crabby-images/7451a/7451a3b591d15107043fbaf040b767bbae725771" alt=""
dva
By sorrycc
dva
dva: react application arch in ant financial
- 81,518