详解redux异步操作实践


Posted in Javascript onAugust 15, 2018

一、redux基础

redux

  1. 通过 dispatch(action) -> 中间件 -> reducer处理数据 -> 改变store -> 使用subscribe()监听store改变更新视图 的方式管理状态
  2. 将所有状态存储在一个store对象里面
  3. reducer为纯函数,而异步操作由于结果的不确定性所以含有副作用,所以需要特殊处理

react-redux

  1. 容器组件,负责管理数据和业务逻辑,不负责UI呈现
  2. UI组件,提供UI呈现,无状态即不使用this.state,状态全部由this.props提供
  3. 由connect生成容器组件,每次store改变会调用connect,connect接收两个参数: mapStateToProps, mapDispatchToProps
  4. mapStateToProps,将状态映射到UI组件的props
  5. mapDispatchToProps,将dispatch方法映射到UI组件的props
  6. Provider组件,使用content API将store从顶层开始传到每一层component供connect使用

二、redux处理异步的中间件

redux-thunk

  1. redux-thunk中间件允许action是一个方法
  2. 中间件收到action后会执行action方法并将结果提供给reducer
  3. action混乱导致不易维护

redux-saga

  1. saga会监听action并基于这个action执行Effects操作
  2. Effects提供灵活的API,包括阻塞、非阻塞调用,取消、等待、race等操作
  3. 方便隔离并执行异步操作,并易于测试

三、redux-request-async-middleware

先从redux文档中的异步action说起,每个接口调用需要dispatch三个同步action,分别是:

  1. 一种通知 reducer 请求开始的 action。对于这种 action,reducer 可能会切换一下 state 中的 isFetching 标记。以此来告诉 UI 来显示加载界面。
  2. 一种通知 reducer 请求成功的 action。对于这种 action,reducer 可能会把接收到的新数据合并到 state 中,并重置 isFetching。UI 则会隐藏加载界面,并显示接收到的数据。
  3. 一种通知 reducer 请求失败的 action。对于这种 action,reducer 可能会重置 isFetching。另外,有些 reducer 会保存这些失败信息,并在 UI 里显示出来。

也就是一个接口发起是这样的

dispatch(fetchPostsRequest(subject));
fetch(url).then(res => {
  dispatch(fetchPostsSuccess(subject, res));
}).catch(e => {
  dispatch(fetchPostsFailure(subject, e));
})

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

  1. 所有的异步请求共用这三个action
  2. 用subject来区分是哪一个请求
  3. 将所有的结果都放到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
};
  1. 和redux-thunk一样,将方法放进action里
  2. 中间件拦截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,
          }
        }
      );
    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,
          }
        }
      );
    default:
      return state;
  }
}
  1. 将结果放入该subject对应下的response,如果错误的话将错误信息放入error当中
  2. isFetching表示当前的请求状态
  3. 另外还加入了当前的状态state和subject信息

将请求进行封装

const request = (subject, model, next) => {
  _dispatch(fetchPostsRequest(subject, model, next));
  return true;
};
  1. 写一个方法来发起FETCH_POSTS_REQUEST action
  2. 也就是说写请求的时候不用再管action这东西了,直接调用request方法

将结果进行封装

const getResponse = state =>
  state
  && state.response !== null
  && state.response;

const getLoading = (states = []) =>
  states.reduce((pre, cur) =>
    pre || (cur && cur.isFetching)
    , false)
  || false;
  1. 可以获取结果和多个请求下loading的状态
  2. 有更多的操作或者格式还可以继续封装,比如列表

使用方法redux-request-async-middleware

四、总结

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

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
一步一步教你写一个jQuery的插件教程(Plugin)
Sep 03 Javascript
jquery获取元素索引值index()示例
Feb 13 Javascript
分享网页检测摇一摇实例代码
Jan 14 Javascript
JS 循环li添加点击事件 (闭包的应用)
Dec 10 Javascript
Angular5中状态管理的实现
Sep 03 Javascript
解决ng-repeat产生的ng-model中取不到值的问题
Oct 02 Javascript
vue+axios 前端实现登录拦截的两种方式(路由拦截、http拦截)
Oct 24 Javascript
深入解析vue 源码目录及构建过程分析
Apr 24 Javascript
微信小程序tab切换可滑动切换导航栏跟随滚动实现代码
Sep 04 Javascript
layui layer select 选择被遮挡的解决方法
Sep 21 Javascript
原生JS实现多条件筛选
Aug 19 Javascript
vue实现书本翻页动画效果实例详解
Apr 08 Vue.js
解决Layui数据表格中checkbox位置不居中的方法
Aug 15 #Javascript
Bootstrap Table列宽拖动的方法
Aug 15 #Javascript
浅谈微信JS-SDK 微信分享接口开发(介绍版)
Aug 15 #Javascript
layui结合form,table的全选、反选v1.0示例讲解
Aug 15 #Javascript
详解微信JS-SDK选择图片遇到的坑
Aug 15 #Javascript
解决layui中的form表单与button的点击事件冲突问题
Aug 15 #Javascript
jQuery仿移动端支付宝键盘的实现代码
Aug 15 #jQuery
You might like
用PHP读取RSS feed的代码
2008/08/01 PHP
PHP基于imap获取邮件实例
2014/11/11 PHP
反射调用private方法实践(php、java)
2015/12/21 PHP
PHP编程快速实现数组去重的方法详解
2017/07/22 PHP
php实现和c#一致的DES加密解密实例
2017/07/24 PHP
javascript parseInt 函数分析(转)
2009/03/21 Javascript
ASP.NET jQuery 实例2 (表单中使用回车在TextBox之间向下移动)
2012/01/13 Javascript
jquery分页插件AmSetPager(自写)
2013/04/15 Javascript
jQuery 三击事件实现代码
2013/09/11 Javascript
Javascript全局变量var与不var的区别深入解析
2013/12/09 Javascript
js读取配置文件自写
2014/02/11 Javascript
jQuery设置和获取HTML、文本和值示例
2014/07/08 Javascript
prototype框架中美元符号$用法分析
2016/01/22 Javascript
jQuery实现点击行选中或取消CheckBox的方法
2016/08/01 Javascript
基于vue的fullpage.js单页滚动插件
2017/03/20 Javascript
angularJS利用ng-repeat遍历二维数组的实例代码
2017/06/03 Javascript
详解微信小程序之一键复制到剪切板
2019/04/24 Javascript
微信小程序把百度地图坐标转换成腾讯地图坐标过程详解
2019/07/10 Javascript
vue实现自定义多选按钮
2020/07/16 Javascript
在Linux中通过Python脚本访问mdb数据库的方法
2015/05/06 Python
Anaconda 离线安装 python 包的操作方法
2018/06/11 Python
Python3.5多进程原理与用法实例分析
2019/04/05 Python
python 杀死自身进程的实现方法
2019/07/01 Python
Python Numpy库datetime类型的处理详解
2019/07/13 Python
python tkinter图形界面代码统计工具(更新)
2019/09/18 Python
python os.path.isfile()因参数问题判断错误的解决
2019/11/29 Python
520使用Python实现“我爱你”表白
2020/05/20 Python
Python实现列表索引批量删除的5种方法
2020/11/16 Python
Shopty西班牙:缝纫机在线销售
2018/01/26 全球购物
售后服务科岗位职责范文
2013/11/13 职场文书
蟋蟀的住宅教学反思
2014/04/26 职场文书
2014年群众路线教育实践活动整改措施
2014/09/24 职场文书
本溪水洞导游词
2015/02/11 职场文书
Java 深入探究讲解简单工厂模式
2022/04/07 Java/Android
Nginx禁止ip访问或非法域名访问
2022/04/07 Servers
Golang实现可重入锁的示例代码
2022/05/25 Golang