前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >React进阶篇(八)react redux

React进阶篇(八)react redux

作者头像
娜姐
发布2020-09-22 10:31:56
1.4K0
发布2020-09-22 10:31:56
举报
文章被收录于专栏:娜姐聊前端

redux是参考Flux设计原则的一个管理数据流的库。

下图为redux的核心运作流程

redux 三大原则:

  • 单一数据源:整个应用的状态都保存在一个对象(store)中。这样我们随时可以取出整个应用的状态进行持久化。
  • 状态只读:不允许直接修改状态(必须通过 dispatch action来修改状态)
  • 纯函数reducer:状态修改为一个纯函数(接受一定的输入,必定会得到一定的输出)完成。
1. Redux核心API

Redux的核心是一个store - 存储状态。

代码语言:javascript
复制
import {createStore} from 'redux';

const store = createStore(reducers, initialState);

store是一个对象,包含4个方法:

  • getState():获取store中当前的状态
  • dispatch(action):分发一个action,并返回这个action,这是唯一能改变store中数据的方式
  • subscribe(listener):注册一个监听者,它在store发生变化时调用
  • replaceReducer(nextReducer):更新当前store里面的reducer,一般只会在开发模式中调用该方法

我们只需要关注 getState() 和 dispatch(action) 即可。

2. 与React绑定使用

使用react-redux库。

3. Redux middleware

Redux 提供了 applyMiddleware方法来加载 middleware。

每个middleware处理一个相对独立的业务需求,通过串联不同的middleware实现变化多样的功能。

代码语言:javascript
复制
export default function applyMiddleware(...middlewares) {

  return (createStore) => (reducer, preloadedState, enhancer) => {
    const store = createStore(reducer, preloadedState, enhancer)
    // 存旧的dispatch 
    let dispatch = store.dispatch  
    let chain = [];

    const middlewareAPI = {
        getState: store.getState,
        // 利用匿名函数包裹dispatch。
        dispatch: (action) => dispatch(action)
    }

    chain = middlewares.map(middleware => middleware(middlewareAPI));
    //生成新的dispatch
    // compose 是函数式编程中的组合,它将 chain 中的所有匿名函数`[f1, f2, ... , fx, ..., fn]`组装成一个新的函数,
    // 即新的dispatch。
    // 多个middleware调用结果为:`dispatch = f1(f2(f3(store.dispatch))))`;
    dispatch = compose(...chain)(store.dispatch);    
    // 返回改写过disptach的store
    return {
      ...store,
      dispatch
    }
  }
}

// 利用applyMiddleware创建新的store。
let newStore = applyMiddleware(mid1, mid2, mid3, ...)(createStore)(reducer, null);
在middleware中调用next(action)和store.dispatch(action)
  • 在middleware中调用next(),可以进入下一个middleware
  • 在middleware中调用store.dispatch(),会跳出middleware流,重新开始。

如上图,分发一个action时,middleware通过next(action)一层层处理和传递action到Redux原生的dispatch。

如果某个middleware使用store.dispatch(action))分发action,会跳出middleware管道,重新再来。

store.dispatch(action)的应用场景

action默认都是同步的。如果是一个异步Action(异步请求),那么需要一个专门处理异步请求的middleware,这是会用到store.dispatch()。

比如redux-thunk:

代码语言:javascript
复制
const thunk = store =>next => action =>
    typeof action === 'function' ? action(store.dispatch, store.getState) : next(action)

异步Acton设计如下:发起异步请求,如果成功,弹出成功弹框,否则,弹出错误弹框。

代码语言:javascript
复制
const getThenShow = (dispatch, getState) =>{
    const url = 'http://xxx.json';

    fetch(url)
        .then(response=>{
            dispatch({
                type: 'SHOW_MESSAGE_SUCCESS',
                message: response.json
            })
        })
        .catch(error=>{
            dispatch({
                type: 'SHOW_MESSAGE_FAIL',
                message: 'error'
            })
        })
}
4. Redux性能优化
  1. 使用react-redux

在使用react-redux的connect函数时,实际上产生了一个无名的React组件类,这个类定制了shouldComponentUpdate函数的实现,实现逻辑时对比这次传递给内层傻瓜组件的props和上次的props。如果props没有变化,那内层组件无需再次渲染。

代码语言:javascript
复制
import {connect} from 'react-redux';

const Foo = ({text})=>(
    <div>{text}</div>
)

const mapStateToProps = (state)=> (
    text: state.text
)

// connect函数返回了一个容器组件,会实现shouldComponentUpdate函数,判断这次的text和上一次text的值是否相等。
export default connect(mapStateToProps)(Foo);

connect函数中采用浅比较,即 value1 == value2

2)用reselect提高数据获取性能

工作原理:只要相关状态没有改变,就直接使用上一次的缓存结果。

reselect把计算过程分为两个步骤:

步骤1: 从输入参数state抽取第一层结果,第一层结果和之前抽取的第一层结果做比较(===比较),如果发现完全相同,就不会进入第二步计算,选择器直接把之前第二部分的运输结果返回。

步骤2: 根据第一层结果计算出选择权需要返回的最终结果。

定义选择器:

代码语言:javascript
复制
import { createSelector } from 'reselect'

const getVisibilityFilter = (state) => state.visibilityFilter
const getTodos = (state) => state.todos

export const getVisibleTodos = createSelector(
  [ getVisibilityFilter, getTodos ],
  (visibilityFilter, todos) => {
    switch (visibilityFilter) {
      case 'SHOW_ALL':
        return todos
      case 'SHOW_COMPLETED':
        return todos.filter(t => t.completed)
      case 'SHOW_ACTIVE':
        return todos.filter(t => !t.completed)
    }
  }
)

使用新定义的选择器:

代码语言:javascript
复制
import {getVisibleTodos} from './selector.js'

const mapStateToProps = (state) => {
    return {
        todos: getVisibleTodos(state);
    }
}

3)Immutable Redux

持久化数据结构,结构共享,惰性操作。

通过Immutable.js创建的对象在任何情况下都无法被修改,这样可以防止由于开发者的粗心导致直接修改Redux的state。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 1. Redux核心API
  • 2. 与React绑定使用
  • 3. Redux middleware
    • 在middleware中调用next(action)和store.dispatch(action)
      • store.dispatch(action)的应用场景
      • 4. Redux性能优化
      相关产品与服务
      容器服务
      腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
      领券
      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档