通常情况下,action 只是一个普通的对象,不能包含异步操作,这导致了很多创建 action 的逻辑只能写在组件中,代码量较多也不便于复用,组件的业务逻辑也不清晰,使用中间件了之后,可以通过actionCreator
异步编写action
,这样代码就会拆分到 actionCreator 中,可维护性大大提高,可以方便于测试、复用,同时 actionCreator 还集成了异步操作中不同的 action 派发机制,减少编码过程中的代码量。
常见的异步库:
基于Promise的异步库:
最简单也是最常用的方法就是使用Redux Thunk
middleware,这样就能用更为复杂或者异步的逻辑书写 action 创建函数。另一个被广泛使用的方法是Redux Saga
,你可以用generator
书写类同步代码,就像在 Redux 应用中使用 “后台线程” 或者 “守护进程”。
在终端中执行下面的命令,安装 redux-thunk
yarn add redux-thunk
创建 store 的时候:
/* store.js */
import {createStore,applyMiddleware} from 'redux'
import reducer from './reducer'
import thunk from 'redux-thunk'
const store=createStore(reducer,applyMiddleware(thunk))
export default store;
reducer.js文件:
/* reducer.js */
const defaultState={ dataList:[] }
const reducer=(state=defaultState,action)=>{
switch(action.type){
case "ADD_LIST_DATA":
return{
dataList:list.concat(action.data)
}
default:
return state;
}
}
export default reducer;
connect.js文件中:
redux-thunk
中间件后,action
可以是一个函数
,在函数中进行异步操作,最终还是执行 dispatch。/* connect.js */
import {connect} from 'react-redux'
import {loadListData} from './actionCreators'
const mapState=(state)=>{
return{ dataList:state.dataList }
}
const mapDispatch=(dispatch)=>{
return{
loadData:(params)=>{
dispatch(loadListData(params))
}
}
}
export default connect(mapState,mapDispatch)
actionCreators.js,请求接口,做异步操作
/* actionCreators.js */
const loadListData=(params)=>{
return(dispatch)=>{
axios.get('/datalist',params).then((rs)=>{
dispatch({type:"ADD_LIST_DATA",data:rs.data})
})
}
}
export { loadListData }
redux-saga
用的是ES6的生成器函数generator
功能,而不是async await
在终端中执行下面的命令,安装 redux-saga
yarn add redux-saga -S
创建 store 的时候:
/* store.js */
import {createStore,applyMiddleware} from 'redux'
import createSagaMiddleware from 'redux-saga'
import reducer from './reducer'
import mySaga from './sagas'
const sagaMiddleware = createSagaMiddleware()
const store = createStore(reducer,applyMiddleware(sagaMiddleware))
sagaMiddleware.run(mySaga)
export default store;
reducer.js文件:
/* reducer.js */
const defaultState={ dataList:[] }
const reducer=(state=defaultState,action)=>{
switch(action.type){
case "ADD_LIST_DATA":
return{
dataList:list.concat(action.data)
}
default:
return state;
}
}
export default reducer;
connect.js文件中:
/* connect.js */
import {connect} from 'react-redux'
const mapState=(state)=>{
return{ dataList:state.dataList }
}
const mapDispatch=(dispatch)=>{
return{
loadData:(params)=>{
dispatch({type:"async_loadListData",...params})
}
}
}
export default connect(mapState,mapDispatch)
sagas.js文件:
/* sagas.js */
import { put, takeEvery } from 'redux-saga/effects'
//请求接口,做异步操作
const loadListData=(params)=> axios.get('/apilist/datalist',params)
function* load(params) {
delete params.type;
let rs = yield loadListData(params);
yield put({type:"ADD_LIST_DATA",data:rs.data})
}
function* sagas(){
yield takeEvery('async_loadListData',load)
}
export default sagas;
redux-saga
用的是generator
语法。thunk
方式,action是个函数
;saga
方式,action还是一个扁平的对象
,会根据这个名字去找 takeEvery()
中的第一个参数,如果一致,则执行 takeEvery() 的第二个参数。