详解redux异步操作示例

#一、redux基础

redux

  • 通过 dispatch(action) -> 中间件 -> reducer处理数据 -> 改变store -> 使用subscribe()监听store改变更新视图 的方式管理状态
  • 将所有状态存储在一个store对象里面
  • reducer为纯函数,而异步操作由于结果的不确定性所以含有副作用,所以需要特殊处理 react-redux 容器组件,负责管理数据和业务逻辑,不负责UI呈现 UI组件,提供UI呈现,无状态即不使用this.state,状态全部由this.props提供 由connect生成容器组件,每次store改变会调用connect,connect接收两个参数: mapStateToProps, mapDispatchToProps mapStateToProps,将状态映射到UI组件的props mapDispatchToProps,将dispatch方法映射到UI组件的props Provider组件,使用content API将store从顶层开始传到每一层component供connect使用 #二、redux处理异步的中间件 redux-thunk redux-thunk中间件允许action是一个方法 中间件收到action后会执行action方法并将结果提供给reducer action混乱导致不易维护 redux-saga saga会监听action并基于这个action执行Effects操作 Effects提供灵活的API,包括阻塞、非阻塞调用,取消、等待、race等操作 方便隔离并执行异步操作,并易于测试 #三、redux-request-async-middleware 先从redux文档中的异步action说起,每个接口调用需要dispatch三个同步action,分别是: 一种通知 reducer 请求开始的 action。对于这种 action,reducer 可能会切换一下 state 中的 isFetching 标记。以此来告诉 UI 来显示加载界面。 一种通知 reducer 请求成功的 action。对于这种 action,reducer 可能会把接收到的新数据合并到 state 中,并重置 isFetching。UI 则会隐藏加载界面,并显示接收到的数据。 一种通知 reducer 请求失败的 action。对于这种 action,reducer 可能会重置 isFetching。另外,有些 reducer 会保存这些失败信息,并在 UI 里显示出来。 也就是一个接口发起是这样的
dispatch(fetchPostsRequest(subject));
fetch(url).then(res => {
  dispatch(fetchPostsSuccess(subject, res));
}).catch(e => {
  dispatch(fetchPostsFailure(subject, e));
})
复制代码

只是将这个操作封装进中间件里,特殊的地方在于:

  • 所有的异步请求共用这三个action
  • 用subject来区分是哪一个请求
  • 将所有的结果都放到store.requests里 中间件源码
export const reduxRequest = store => next => action => {
  let result = next(action);
  let { type, subject, model } = action;
  let _next = action.next;
  if(type === FETCH_POSTS_REQUEST) {
    model().then(response => {
      _next && _next(response);
      store.dispatch(fetchPostsSuccess(subject, response));
    }).catch(error => {
      console.error(error);
      store.dispatch(fetchPostsFailure(subject, error));
    });
  }
  return result
};//欢迎加入全栈开发交流圈一起学习交流:864305860
复制代码
  • 和redux-thunk一样,将方法放进action里
  • 中间件拦截FETCH_POSTS_REQUEST action,并进行异步处理 reducer源码
export const requests = (state = {}, action) => {
  switch (action.type) {
    case FETCH_POSTS_REQUEST:
      return assign({},
        state,
        {
          [action.subject]: {
            isFetching: true,
            state: 'loading',
            subject: action.subject,
            response: null,
            error: null,
          }
        }
      );
    case FETCH_POSTS_FAILURE:
      return assign({},
        state,
        {
          [action.subject]: {
            isFetching: false,
            state: 'error',
            subject: action.subject,
            response: state[action.subject].response,
            error: action.error,
          }//欢迎加入全栈开发交流圈一起学习交流:864305860
        }//面向1-3年前端人员
      );//帮助突破技术瓶颈,提升思维能力
    case FETCH_POSTS_SUCCESS:
      return assign({},
        state,
        {
          [action.subject]: {
            isFetching: false,
            state: 'success',
            subject: action.subject,
            response: action.response,
          }
        }
      );
    case FETCH_POSTS_CLEAR:
      return assign({},
        state,
        {
          [action.subject]: {
            isFetching: false,
            state: 'cleared',
            subject: null,
            response: null,
            error: null,
          }//欢迎加入全栈开发交流圈一起学习交流:864305860
        }//面向1-3年前端人员
      );//帮助突破技术瓶颈,提升思维能力
    default:
      return state;
  }
}
复制代码
  • 将结果放入该subject对应下的response,如果错误的话将错误信息放入error当中
  • isFetching表示当前的请求状态
  • 另外还加入了当前的状态state和subject信息 将请求进行封装
const request = (subject, model, next) => {
  _dispatch(fetchPostsRequest(subject, model, next));
  return true;
};
复制代码
  • 写一个方法来发起FETCH_POSTS_REQUEST action
  • 也就是说写请求的时候不用再管action这东西了,直接调用request方法 将结果进行封装
const getResponse = state =>
  state
  && state.response !== null
  && state.response;
 
const getLoading = (states = []) =>
  states.reduce((pre, cur) =>
    pre || (cur && cur.isFetching)
    , false)//欢迎加入全栈开发交流圈一起学习交流:864305860
  || false;//面向1-3年前端人员
//帮助突破技术瓶颈,提升思维能力
复制代码
  1. 可以获取结果和多个请求下loading的状态
  2. 有更多的操作或者格式还可以继续封装,比如列表

四、总结

  1. 使用了redux来进行状态管理,而并不需要编写redux那一套复杂逻辑,最大程度的减少异步操作的复杂度
  2. 适用于前端通过接口来处理和存储数据的项目
  3. 接口由redux处理,而视图组件由内部state来处理,而外部只暴露简单的接口来进行操作,分离业务层和视图层
  4. 对比react 16.3 new content API,redux的优势在于热插播的中间件和纯函数reducer写法

本次给大家推荐一个免费的学习群,里面概括移动应用网站开发,css,html,webpack,vue node angular以及面试资源等。 对web开发技术感兴趣的同学,欢迎加入Q群:864305860,不管你是小白还是大牛我都欢迎,还有大牛整理的一套高效率学习路线和教程与您免费分享,同时每天更新视频资料。 最后,祝大家早日学有所成,拿到满意offer,快速升职加薪,走上人生巅峰。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Redux中,异步操作可以通过中间件来处理。在Redux中,中间件是一种函数,它可以拦截派发到Store中的Action,并在必要时对其进行操作。常见的Redux中间件有Redux Thunk、Redux Saga和Redux Observable等。 其中,Redux Thunk是Redux官方提供的中间件之一,它允许Action Creator返回一个函数而不是一个对象。在这个函数中,我们可以执行异步操作并在操作完成后再派发一个Action来更新应用程序状态。 例如,我们可以编写一个异步Action Creator来获取用户列表: ```javascript const fetchUsers = () => { return (dispatch) => { dispatch({ type: 'FETCH_USERS_REQUEST' }); return fetch('/users') .then(response => response.json()) .then(data => { dispatch({ type: 'FETCH_USERS_SUCCESS', payload: data }); }) .catch(error => { dispatch({ type: 'FETCH_USERS_FAILURE', payload: error }); }); }; }; ``` 在上面的代码中,我们通过返回一个函数来处理异步操作。在函数中,我们首先派发一个FETCH_USERS_REQUEST Action来表示开始获取用户列表,然后执行异步操作。当异步操作完成后,我们再派发一个FETCH_USERS_SUCCESS Action来更新应用程序状态,并将获取到的用户列表作为payload传递给Action。如果异步操作失败,我们将派发一个FETCH_USERS_FAILURE Action来通知应用程序。 通过这种方式,我们可以在Redux中处理异步操作,并保持应用程序的状态可预测性和一致性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值