redux 源码解读

前言

redux并不局限于flux与react。redux 自身保持简洁以便适配各种场景,让社区发展出各种 redux-* 中间件或者插件,从而形成它自己的生态系统。

主要关系

  • reducer 声明了state的初始值,以及当前state接受action对象之后处理为new state的逻辑。

  • createStore接受reducer作为参数,返回产生的store,store其实是一个含有state的闭包,以及提供将action分发给reducer的dispatch方法

  • applyMiddlewares方法接受n个middlewares作为参数返回一个用于渲染creatorStore函数方法

  • applyMiddleware可以向actionCreator提供store.dispatch以及getState方法,用以增强actionCreator的能力

store主要包含以下三个核心方法:

  • subscribe 注册store更新之后的回调函数

  • getState 获取store当前state的引用,切记直接修改返回的结果

  • dispatch 将action按顺序经过各middle处理后派发给reducer

action 流程图

createStore

createStore是根据reducer中的规则创建store的方法

特性

  1. 提供dispatch

  2. subscribe

  3. getState // getState拿到的是state的引用!不要直接修改

  4. 提供初始值initialState

源码

//此处为示意,不是 redux 的源码本身
export default createStore(reducer,initialState) {
    //闭包私有变量
    let currentState = initialState
    let currentReducer = reducer
    let listeners = []

    //返回一个包含可访问闭包变量的公有方法
    return {
        getState() {
            return currentState //返回当前 state
        },subscribe(listener) {
            let index = listeners.length
            listeners.push(listener) //缓存 listener
            return () => listeners.splice(i,1) //返回删除该 listener 的函数
        },dispatch(action) {
            //更新 currentState
            currentState = currentReducer(currentState,action)
            // 可以看到这里并没有用到eventEmitter等
            listeners.slice().forEach(listener => listener())
            return action //返回 action 对象
        }
    }
}

action

action有以下特点:

  • pure object

  • 描述reducer响应的事件类型

  • 携带所需要的数据

actionCreator

用于描述action的dispatch的逻辑。

  • action的重用

  • 数据的预处理

  • action的特殊处理逻辑

reducer

reducer应该是是一个无副作用函数,以当前的state以及action为参数,返回新的state。
每次返回一个新State的好处是在shouldComponentUpdate过程中可以使用高性能的shallow equal。

  1. pure function

  2. 接受initialState

  3. don't modify the state!!!

//reducer 接受两个参数,全局数据对象 state 以及 action 函数返回的 action 对象
//返回新的全局数据对象 new state
export default (state,action) => {
    switch (action.type) {
        case A:
        return handleA(state)
        case B:
        return handleB(state)
        case C:
        return handleC(state)
        default:
        return state //如果没有匹配上就直接返回原 state
    }
}

combineReducers

一个reducer map转换为一个reducer。方便对复杂的reducer进行功能拆分。

problem

  1. state 结构太复杂

  2. 希望根据对应的component进行维护

how to use

var reducers = {
    todos: (state,action) { //预期此处的 state 参数是全局 state.todos 属性
        switch (action.type) {...} //返回的 new state 更新到全局 state.todos 属性中
    },activeFilter: (state,action) { //预期拿到 state.activeFilter 作为此处的 state
        switch (action.type) {...} //new state 更新到全局 state.activeFilter 属性中
    }
}

//返回一个 rootReducer 函数
//在内部将 reducers.todos 函数的返回值,挂到 state.todos 中
//在内部将 reducers.activeFilter 函数的返回值,挂到 state.activeFilter 中
var rootReducer = combineReducers(reducers)

源码

//combination 函数是 combineReducers(reducers) 的返回值,它是真正的 rootReducer
//finalReducers 是 combineReducers(reducers) 的 reducers 对象去掉非函数属性的产物
 //mapValue 把 finalReducers 对象里的函数,映射到相同 key 值的新对象中
function combination(state = defaultState,action) {
    var finalState = mapValues(finalReducers,(reducer,key) => {
      var newState = reducer(state[key],action); //这里调用子 reducer
      if (typeof newState === 'undefined') {
        throw new Error(getErrorMessage(key,action));
      }
      return newState; //返回新的子 state
    });
    //...省略一些业务无关的代码
    return finalState; //返回新 state
 };

function mapValues(obj,fn) {
  return Object.keys(obj).reduce((result,key) => {
    result[key] = fn(obj[key],key);
    return result;
  },{});
}

applyMiddleWares

problem

  • 异步action

  • promise

  • 个性化 action 响应

  • log

描述

接受 middleWares 将 store 修饰为使用了 middlwares 的 store,其实是用被高阶函数修饰过的dispatch替换掉了原来的dispatch。

usage

var craeteStoreWithMiddleWare = applyMiddleWare(thunk)(createStore);
//redux-thunk
export default function thunkMiddleware({ dispatch,getState }) {
  return next => action =>
    typeof action === 'function' ? // action 居然是函数而不是 plain object?
      action(dispatch,getState) : //在中间件里消化掉,让该函数控制 dispatch 时机
      next(action); //否则调用 next 让其他中间件处理其他类型的 action
}

源码

这里的composeMiddleware可能不是很好理解,这里有一个简单的例子方便大家理解。http://jsbin.com/xalunadofa/1/edit?js,console。 compose可以理解为倒叙一层层打包的过程,因此最后调用composedFunction的时候会顺序进入各个middlewares。

function applyMiddleware(...middlewares) {
  return next => (...args) => {
    const store = next(...args);
    const middleware = composeMiddleware(...middlewares);

      // dispatch 被middlWare修饰
    function dispatch(action) {
      const methods = {
        dispatch,getState: store.getState
      };

      return compose(
        middleware(methods),store.dispatch
      )(action);
    }

      // 返回新的store dispatch被新的dispatch替代
    return {
      ...store,dispatch
    };
  };
}

bindActionCreator

源码

//将 actionCreator 跟 dispatch 绑定在一起
let bindActionCreator => (actionCreator,dispatch) {
  return (...args) => dispatch(actionCreator(...args));
}

function bindActionCreators(actionCreators,dispatch) {
  if (typeof actionCreators === 'function') { //如果是单个 actionCreator,绑定一词
    return bindActionCreator(actionCreators,dispatch);
  }
  //返回一个改造过的「函数组合」
  return mapValues(actionCreators,actionCreator =>
    bindActionCreator(actionCreator,dispatch)
  )
}

connector

connector 接受mapStatetoProps,mapdispatchToProps,Component 三个参数,返回一个能够自动关联store中state以及dispatch事件的smart component

由于connector代码过长,只对重要的几个函数进行说明。
connetor函数接受的两个参数指明从store的state中挑选哪些作为props,以及将哪些actionCreator绑定到porps中。
订阅store的change事件,当store更新时计算新的state,与旧state进行浅对比,如果不同则更新state,并render,否则不进行render。

// 根据从store中select的state以及dispatch绑定的actionCreator计算新的props
   computeNextState(props = this.props) {
     return computeNextState(
       this.stateProps,this.dispatchProps,props
     );
   }

    // 与旧值进行shallow equal
   updateState(props = this.props) {
     const nextState = this.computeNextState(props);
     if (!shallowEqual(nextState,this.state.props)) {
       this.setState({
         props: nextState
       });
     }
   }

    // 订阅change事件
   trySubscribe() {
     if (shouldSubscribe && !this.unsubscribe) {
       this.unsubscribe = this.store.subscribe(::this.handleChange);
       this.handleChange();
     }
   }

   tryUnsubscribe() {
     if (this.unsubscribe) {
       this.unsubscribe();
       this.unsubscribe = null;
     }
   }

   componentDidMount() {
     this.trySubscribe();
   }


   componentwillUnmount() {
     this.tryUnsubscribe();
   }

   handleChange() {
     if (!this.unsubscribe) {
       return;
     }

     if (this.updateStateProps()) {
       this.updateState();
     }
   }

结语

欢迎大家发起pr完善文档,进行讨论。

参考资料

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


react 中的高阶组件主要是对于 hooks 之前的类组件来说的,如果组件之中有复用的代码,需要重新创建一个父类,父类中存储公共代码,返回子类,同时把公用属性...
我们上一节了解了组件的更新机制,但是只是停留在表层上,例如我们的 setState 函数式同步执行的,我们的事件处理直接绑定在了 dom 元素上,这些都跟 re...
我们上一节了解了 react 的虚拟 dom 的格式,如何把虚拟 dom 转为真实 dom 进行挂载。其实函数是组件和类组件也是在这个基础上包裹了一层,一个是调...
react 本身提供了克隆组件的方法,但是平时开发中可能很少使用,可能是不了解。我公司的项目就没有使用,但是在很多三方库中都有使用。本小节我们来学习下如果使用该...
mobx 是一个简单可扩展的状态管理库,中文官网链接。小编在接触 react 就一直使用 mobx 库,上手简单不复杂。
我们在平常的开发中不可避免的会有很多列表渲染逻辑,在 pc 端可以使用分页进行渲染数限制,在移动端可以使用下拉加载更多。但是对于大量的列表渲染,特别像有实时数据...
本小节开始前,我们先答复下一个同学的问题。上一小节发布后,有小伙伴后台来信问到:‘小编你只讲了类组件中怎么使用 ref,那在函数式组件中怎么使用呢?’。确实我们...
上一小节我们了解了固定高度的滚动列表实现,因为是固定高度所以容器总高度和每个元素的 size、offset 很容易得到,这种场景也适合我们常见的大部分场景,例如...
上一小节我们处理了 setState 的批量更新机制,但是我们有两个遗漏点,一个是源码中的 setState 可以传入函数,同时 setState 可以传入第二...
我们知道 react 进行页面渲染或者刷新的时候,会从根节点到子节点全部执行一遍,即使子组件中没有状态的改变,也会执行。这就造成了性能不必要的浪费。之前我们了解...
在平时工作中的某些场景下,你可能想在整个组件树中传递数据,但却不想手动地通过 props 属性在每一层传递属性,contextAPI 应用而生。
楼主最近入职新单位了,恰好新单位使用的技术栈是 react,因为之前一直进行的是 vue2/vue3 和小程序开发,对于这些技术栈实现机制也有一些了解,最少面试...
我们上一节了了解了函数式组件和类组件的处理方式,本质就是处理基于 babel 处理后的 type 类型,最后还是要处理虚拟 dom。本小节我们学习下组件的更新机...
前面几节我们学习了解了 react 的渲染机制和生命周期,本节我们正式进入基本面试必考的核心地带 -- diff 算法,了解如何优化和复用 dom 操作的,还有...
我们在之前已经学习过 react 生命周期,但是在 16 版本中 will 类的生命周期进行了废除,虽然依然可以用,但是需要加上 UNSAFE 开头,表示是不安...
上一小节我们学习了 react 中类组件的优化方式,对于 hooks 为主流的函数式编程,react 也提供了优化方式 memo 方法,本小节我们来了解下它的用...
开源不易,感谢你的支持,❤ star me if you like concent ^_^
hel-micro,模块联邦sdk化,免构建、热更新、工具链无关的微模块方案 ,欢迎关注与了解
本文主题围绕concent的setup和react的五把钩子来展开,既然提到了setup就离不开composition api这个关键词,准确的说setup是由...
ReactsetState的执行是异步还是同步官方文档是这么说的setState()doesnotalwaysimmediatelyupdatethecomponent.Itmaybatchordefertheupdateuntillater.Thismakesreadingthis.staterightaftercallingsetState()apotentialpitfall.Instead,usecom