添加链接
link管理
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接

Redux 源码解析 - 从零实现一个 Redux

May 20, 2021

1. Redux 简单介绍

Redux 是 JavaScript 应用状态容器,提供可预测化的状态管理。

可以让你构建一致化的应用,运行于不同的环境(客户端、服务器、原生应用),并且易于测试。不仅于此,它还提供 超爽的开发体验,比如有一个 时间旅行调试器可以编辑后实时预览

2. 为什么要用 redux,什么情况下需要用到 redux

曾经有人说过这样一句话。

"如果你不知道是否需要 Redux,那就是不需要它。"

Redux 的创造者 Dan Abramov 又补充了一句。

"只有遇到 React 实在解决不了的问题,你才需要 Redux 。"

如果一个应用不是很复杂,可以通过 react 的内部 state 就满足对数据的管理,那就没必要使用 redux。

因为使用 redux 每次写代码会额外增加代码的编写以及可能需要多创建多几个文件。

那什么时候适合用 redux?

  • 组件树庞大,不同的组件节点需要共享状态
  • 某个组件需要去修改全局的状态或修改其他组件的状态
  • 与后端服务器有比较多的数据交互,并且组件视图依赖后端返回的数据
  • 使用 redux 带来什么好处?

  • redux 让 state 的变化变得可预测,因为修改 state 只能通过 dispatch 一个 action,这个过程是可监控的,在开发环境中,结合 redux-devtools 还可以实现时间旅行、录制、重放等
  • redux 将 state 统一管理,会使代码更加有规律,易于维护管理。
  • 3. redux 基础知识

    3.1 redux API

  • 顶级暴露的 API
  • createStore(reducer, [preloadedState], [enhancer])
  • combineReducers(reducers)
  • applyMiddleware(...middlewares)
  • bindActionCreators(actionCreators, dispatch)
  • compose(...functions)
  • Store API
  • getState()
  • dispatch(action)
  • subscribe(listener)
  • replaceReducer(nextReducer)
  • 3.1.1 createStore

    createStore 用于创建一个 store 实例,创建时,第一个参数会传入一个 reducer 函数, store 保存了 state 数据,传入的 reducer 函数定义了修改 state 的规则。

    3.1.2 store.getState

    store . getSate ( ) 返回 store 内部的 state 数据, state 数据在外部无法直接访问,必须通过 getState 方法获取

    3.1.3 store.dispatch

    store . dispatch ( ) 派发一个 action action 是一个对象,一般至少包含一个 type 字段,例如: { type : ' add ' } ,除了 type 字段外,也可以包含其他数据,type 用于匹配 reducer 中的修改规则,其他数据用于更新 state。

    想要修改 state 中的数据,必须通过 dispatch

    3.1.4 store.subscribe

    store . subscribe 用于订阅 store 的数据变化,监听函数会在 dispatch 调用时执行

    3.2 redux flow

    redux 与 react 结合使用的数据流:

  • 在 react 组件中调用 dispatch ( action ) ,派发一个 action action 是一个描述 “发生了什么” 的普通对象,例如: { type : ' add ' , value : 1 } ,这个action 对象可以理解为 “增加数值 1”
  • store 取到 action 后,调用构建 store 传入到 reducer 函数,并且传入当前的 state action
  • reducer 取到 state action 后,会通过 action . type 匹配到修改 state 的规则,然后修改并返回新的 state
  • store 保存新的 state,然后所有订阅了 store . subscribe ( ) 的监听器都被调用,并通过 store . getState ( ) 获取新的 state,重新渲染组件。
  • 4. redux 使用

    4.1 reducer 介绍

    创建 store 的时候,需要传入一个 reducer 函数。

    reducer = ( state , action ) = > { }

    Reducer (也称为 reducing function ) 函数接受两个参数:

  • state: 之前累积运算的结果和当前被累积的值
  • action: 描述了发生什么事情的一个对象,一般带有 type 字段。例如: { type : ' ADD ' }
  • 返回值是一个新的累积结果(state)。

    Reducers 指定了应用状态的变化如何响应 actions 并发送到 store 的,记住 actions 只是描述了 有事情发生了 这一事实,并没有描述应用如何更新 state。

    reducer 是一个纯函数,即接收的参数不变的情况下,返回值也不会发生变化。

    永远不要 在 reducer 里做这些操作:

  • 修改传入参数;
  • 执行有副作用的操作,如 API 请求和路由跳转;
  • 调用非纯函数,如 Date . now ( ) Math . random ( )
  • reducer 一定要保持纯净。 只要传入参数相同,返回计算得到的下一个 state 就一定相同。没有特殊情况、没有副作用,没有 API 请求、没有变量修改,单纯执行计算。

    4.2 创建一个 store

    安装 redux : npm install redux yarn add redux

    src/store/index.js

    import { createStore } from 'redux';
    // 定义 reducer 函数
    const reducer = (state = 0, action) => {
      switch (action.type) {
        case 'add':
          return state + 1;
        case 'minus':
          return state - 1;
        default:
          return state;
    // 使用 createStore ,传入 reducer 函数,生成一个 store
    const store = createStore(reducer);
    export default store;

    4.3 使用 store

    src/pages/ReduxPage.js

    import React, { Component } from 'react';
    import store from '../store';
    export default class ReduxPage extends Component {
      componentDidMount() {
        // 到目前为止,通过 dispath 一个 action 修改 state 后,页面并不会自动更新
        // 在没有使用其他库配合前,暂时使用这种方式更新界面,订阅 store 变化,然后强制更新页面
        this.unsubscribe = store.subscribe(() => {
          this.forceUpdate();
        });
      add = () => {
        store.dispatch({ type: 'ADD' });
      minus = () => {
        store.dispatch({ type: 'MINUS' });
      render() {
        return (
            ReduxPage
            <div>count: {store.getState()}</div>
            <button onClick={this.add}>add</button>
            <button onClick={this.minus}>minus</button>
            <button onClick={()=> this.unsubscribe()}>unsubscribe</button>
          </div>
    

    src/App.js

    import React from 'react';
    import ReduxPage from './page/ReduxPage';
    function App() {
      return (
        <div className="App">
          <ReduxPage />
        </div>
    export default App;

    到此就已经可以简单的使用 redux 进行状态管理

    5.1 createStore 实现

    创建 redux 并暴露 createStore 方法

    src/redux/index.js

    import createStore from './createStore'
    export {
      createStore
    
    

    createStore 创建了一个 store 实例,这个实例包含了 getStatesubscribedispatch 等方法

    export default function createStore(reducer) {
      function getSate() {}
      function subscribe() {}
      function dispatch() {}
      return {
        getSate,
        subscribe,
        dispatch,
    
    

    getState 只需要返回当前等 state

    export default function createStore(reducer) {
      // 定义 currentState,保存当前的 state
      let currentState;
      function getSate() {
        // 调用 getState 时,返回当前的 state
        return currentState;
      function subscribe() {}
      function dispatch() {}
      return {
        getSate,
        subscribe,
        dispatch,
    
    

    subscribe,将订阅的回调函数保存到回调函数数组中

    export default function createStore(reducer) {
      let currentState;
      // 定义 currentListeners,保存订阅的回调函数
      let lisenters = [];
      function getSate() {
        return currentState;
      function subscribe(listener) {
        // 将回调函数保存到回调函数数组中
        lisenters.push(listener)
        // 返回一个取消订阅的函数
        return function unsubscribe() {
          const index = lisenters.indexOf(lisenter);
          lisenters.splice(index, 1);
      function dispatch() {}
      return {
        getSate,
        subscribe,
        dispatch,
    
    

    dispatch 一个 action (dispatch(action))是修改 state 的唯一方法。

    dispatch 方法会调用创建 store 时传入的 reducer 函数,并把当前的 state 和 action 传入。

    最后会循环调用 subscribe 方法收集的回调函数。

    export default function createStore(reducer) {
      let currentState;
      let lisenters = [];
      function getSate() {
        return currentState;
      function subscribe(listener) {
        lisenters.push(listener);
        return function unsubscribe() {
          const index = lisenters.indexOf(lisenter);
          lisenters.splice(index, 1);
      function dispatch(action) {
        // 调用 reducer 函数,修改当前的 state
        currentState = reducer(currentState, action)
        // 循环调用回调函数
        for (let i = 0; i < lisenters.length; i++) {
          const lisenter = lisenters[i];
          lisenter()
      return {
        getSate,
        subscribe,
        dispatch,
    
    
    export default function createStore(reducer) {
      let currentState;
      let lisenters = [];
      function getState() {/*略*/}
      function subscribe(listener) {/*略*/}
      function dispatch(action) {/*略*/}
      // 执行一次初始化的 dispatch,这样可以保证整个 state 树都拥有初始状态值,这样在定义 reducer 函数时定义都初始 state 才会生效。
      dispatch({ type: '@@redux/INIT' });
      return {
        getState,
        subscribe,
        dispatch,
    

    到这里,已经实现了 redux 的基础功能,除此之外,上述的几个方法中,还需要做一些参数兼容性处理,边缘情况的处理等。

    此时可以修改上面的例子中的代码,把 redux 改为自己实现的 redux(src/redux/index.js),修改后功能正常使用。

    目前这里 action 还只能支持 js 的普通对象(plain object),action 不支持传入一个函数(异步)。如果要实现异步操作,要传入一个值为函数的 action 还需要借助中间件。

    如何实现异步操作?

    到目前为止,使用 redux 还只能使用普通对象作为 action,如果需要异步的更新 state,例如实现动态从后端获取数据,然后修改 state,还无法实现。

    import React, { Component } from 'react';
    import store from '../store';
    export default class ReduxPage extends Component {
      componentDidMount() {
        store.subscribe(() => {
          this.forceUpdate();
        });
      add = () => {
        store.dispatch({ type: 'ADD' });
      minus = () => {
        store.dispatch({ type: 'MINUS' });
    	// 传入 action 为函数,进行异步操作
      asyncAdd = () => {
        store.dispatch(() => {
          setTimeout(() => {
            store.dispatch({ type: 'ADD' });
          });
        });
      render() {
        return (
            <div>count: {store.getState()}</div>
            <button onClick={this.add}>add</button>
            <button onClick={this.minus}>minus</button>
            <button onClick={this.asyncAdd}>asyncAdd</button>
          </div>
    

    要实现异步操作,需要接入 redux 中间件

    Redux 只是纯粹的状态管理器,默认只支持同步,实现异步任务比如延迟、网络请求等,需要中间件的支持

    store 是 redux 的核心内容,除了 store 相关都内容,redux 还提供了其他一些 api,用于扩展、接入其他库,实现更强大的功能。其中 applyMiddlewares 用于中间件的接入应用。

    中间件的执行在调用 dispatch 到最终 reducer 函数修改 state 的这个过程之间。

    接下来试用一下最简单的 thunk 和 logger 中间件

    src/store/index.js

    import { createStore, applyMiddleware } from 'redux';
    // import { createStore } from '../redux';
    // 引入中间件
    import thunk from 'redux-thunk';
    import logger from 'redux-logger';
    const reducer = function (state = 0, action) {
      switch (action.type) {
        case 'ADD':
          return state + 1;
        case 'MINUS':
          return state - 1;
        default:
          return state;
    // 使用 applyMiddleware 接入 logger,thunk 中间件
    const store = createStore(reducer, applyMiddleware(logger, thunk));
    export default store;

    中间件是按照一定的顺序执行的,调整中间件的位置,可能会得到不同的执行结果。

    例如上面例子,logger 放在 thunk 前面,可以打印到异步的action 的日子,如果 logger 放到 thunk 后面,就无法打印出异步的action 的信息,因为 thunk 的逻辑代码中,如果 action 是函数,就会执行该函数,直接跳过后续中间件。这个在实际开发中可能要了解并注意一下。

    7. redux 中间件实现

    实际上中间件是对原始的 dispatch 做了增强、扩展,在 dispatch 外面又包了一层函数,执行中间件本身的功能,执行完中间件本身的功能后,再去调用原始的 dispatch。

    中间件可能是一个,也可能是无数个,要保证这些中间件都能有效并且有序的执行,就需要一个好的方法将中间件串联起来。

    7.1 实现 compose 函数

    compose 函数实现将中间件串起来,一个接一个执行,并将上一个中间件的执行结果,传过下个中间件。

    实现这样的效果 compose(f1, f2, f3)(...args) => f1(f2(f3(...args)))

    compose 的实现使用到 Array.prototype.reduce() 方法。

    src/redux/compose.js

    export default function compose(...funcs) {
      // funcs 长度为 0 的话,返回一个默认的函数
      if(funcs.length === 0) {
        return arg => arg
      // funcs 长度为 1 的话,直接返回该函数
      if(funcs.length === 1) {
        return funcs[0]
      // 使用 reduce 将函数串联起来
      return funcs.reduce((a,b) => (...args) => a(b(...args)))
      // 下面是将上面这行代码的箭头函数拆开,方便理解,要想理解这段代码,必须先理解 reduce 的使用。
      // 例如:funcs 是 [logger,thunk],则最终会得到的是 logger(thunk(..args))
      // return funcs.reduce(function(a, b) {
      // 	 return function(...args) {
      //     return a(b(...args))
      //   }
    	// })
    

    7.2 实现 applyMiddleware 函数

    applyMiddleware 函数传入中间件,并返回一个增强函数,该函数会对 createStore 函数进行增强。

    src/redux/applyMiddleware.js

    import compose from './compose';
    export default function applyMiddleware(...middlewares) {
      // 返回一个函数,这个函数会接收 createStore 作为参数,并对其进行增强,返回一个增强后的 createStore 函数
      return (createStore) => (reducer) => {
        // 使用原始的 createStore 函数,创建好 store
        let store = createStore(reducer);
        let dispatch = () => {
          throw new Error(
            '不允许在构建中间件时调用 dispatch,其他中间件不会应用此 dispatch。'
        // 提供给中间件的 API,中间件在构建时可以用到这些 api
        const middlewareAPI = {
          getState: store.getState,
          dispatch: (action, ...args) => dispatch(action, ...args),
        // 调用中间件函数(对所有中间件进行构建)
        const chain = middlewares.map((middleware) => middleware(middlewareAPI));
        // 使用 compose 将中间件数组串起来
        dispatch = compose(...chain)(store.dispatch);
        return {
          ...store,
          dispatch,
    

    这里 let dispatch 定义了一个函数,最后才给 dispatch 赋值为“增强” 后的 dispatch, 是为了防止在构建中间件的时候就调用 dispatch。

    src/redux/createStore.js

    修改 createStore 函数增加一个 enhancer 参数,用于增强 createStore

    export default function createStore(reducer, enhancer) {
      if (typeof enhancer !== 'undefined') {
        if (typeof enhancer !== 'function') {
          throw new Error('enhancer 必须是一个函数');
        return enhancer(createStore)(reducer);
      /*略*/
    

    当使用 applyMiddleware 函数接入中间件的时候增加 store 功能的时候,enhancer 就是 applyMiddleware

    中间件一共3层函数,

    第一层函数,是中间件的构建函数,构建时会传入 getState 和 dispatch 方法,中间件的初始化也会在这里执行,使中间件在执行时可以用到这两个方法。

    第二层函数,主要用于将中间件串连起来

    第三层函数,是该中间件要扩展的功能的主要逻辑代码实现

    7.3.1 实现 redux-thunk

    // 构建中间件的时候,传入 getState, dispatch,使中间件可以用这两个方法
    const thunk = ({ getState, dispatch }) => {
      // 第二层是 compose 的时候生成的一层包一层的函数,其中 next 就是下一层中间件,最后一个 next 就是原始的 dispatch
      return (next) => {
        // 中间件主要逻辑代码
        return (action) => {
          if (typeof action === 'function') {
            return action(dispatch, getState);
          return next(action);
    
    

    单纯依靠 redux 的话,只能通过 subscribe 订阅数据的变化,手动更新界面,并且每次都需要通过 getState 获取最新的数据。

    这样操作起来比较麻烦。

    要更好的将 react 跟 redux 结合起来,就需要借助像 react-redux 这样的库。

    8.1 react-redux api

    react-redux 提供了两个 api:Providerconnect

  • <Provider store>
  • connect([mapStateToProps], [mapDispatchToProps], [mergeProps], [options])
  • Provider 使组件层级中的 connect() 方法都能够获得 Redux store,正常情况下,你的根组件应该嵌套在 <Provider> 中才能使用 connect() 方法。
  • connect 顾名思义,就是连接 Redux store 与组件。它是一个高阶组件(HOC),传入一个组件,并且返回一个新的组件,扩展原来的组件使原来组件可以获取到 sotre 中的数据与变更数据的方法。连接操作不会改变原来的组件类。而是返回一个新的已与 Redux store 连接的组件类。
  • 8.2 react-redux 使用

    将上面例子改成使用 react-redux

    src/index.js

    import React from 'react';
    import ReactDOM from 'react-dom';
    import App from './App';
    import { Provider } from './react-redux';
    import store from './store'
    // 使用 Provider 为后代组件提供 store,使组件层级中的 connect() 方法都能够获得 Redux store
    ReactDOM.render(
      <Provider store={store}>
        <App />
      </Provider>,
      document.getElementById('root')
    

    src/page/ReactReduxPage.js

    import React, { Component } from 'react';
    import { connect } from '../react-redux';
    class ReactReduxPage extends Component {
      add = () => {
        this.props.dispatch({ type: 'ADD' });
      minus = () => {
        this.props.dispatch({ type: 'MINUS' });
      asyncAdd = () => {
        this.props.dispatch((dispatch) => {
          setTimeout(() => {
            dispatch({ type: 'ADD' });
          }, 1000);
        });
      render() {
        return (
            <div>count: {this.props.count}</div>
            {/* <button onClick={this.add}>add</button> */}
            <button onClick={this.props.add}>add</button>
            <button onClick={this.minus}>minus</button>
            <button onClick={this.asyncAdd}>asyncAdd</button>
          </div>
    // mapStateToProps 用于将 redux 的 state 传给 组件
    // mapDispatchToProps 用于将 dispatch 与 action 封装成方法,再传给组件,方便组件里修改 state,而不用写大量 dispatch(xxxx)
    const mapStateToProps = (state) => ({ count: state });
    // mapDispatchToProps 传一个对象的使用方式
    const mapDispatchToProps = {
      add: () => ({ type: 'ADD' }),
    // mapDispatchToProps 传一个函数的使用方式,下面代码与上面传对象的方式实现同样的功能
    // const mapDispatchToProps = (dispatch) => {
    //   return {
    //     dispatch,
    //     add: () => dispatch({ type: 'ADD' }),
    //   };
    // };
    // 使用 connect 方法连接组件与 store
    // 传入的 mapStateToProps 和 mapDispatchToProps 定义了需要传递给组件的 state 与 dispatch
    export default connect(mapStateToProps, mapDispatchToProps)(ReactReduxPage);
    
    

    这里 Context 用于将 redux 中的 state 传递到各个组件,使组件可以方便的使用 redux 的 state 中的数据。

    在一个典型的 React 应用中,数据是通过 props 属性自上而下(由父及子)进行传递的,但这种做法对于某些类型的属性而言是极其繁琐的(例如:地区偏好,UI 主题),这些属性是应用程序中许多组件都需要的。Context 提供了一种在组件之间共享此类值的方式,而不必显式地通过组件树的逐层传递 props。

    Context 在日常开发中用的比较少,在第三方库中用到比较多。react-redux 中就用到这个功能。

    Context 的使用

    // Context 可以让我们无须明确地传遍每一个组件,就能将值深入传递进组件树。
    // 为当前的 theme 创建一个 context(“light”为默认值)。
    const ThemeContext = React.createContext('light');
    class App extends React.Component {
      render() {
        // 使用一个 Provider 来将当前的 theme 传递给以下的组件树。
        // 无论多深,任何组件都能读取这个值。
        // 在这个例子中,我们将 “dark” 作为当前的值传递下去。
        return (
          <ThemeContext.Provider value="dark">
            <Toolbar />
          </ThemeContext.Provider>
    // 中间的组件再也不必指明往下传递 theme 了。
    function Toolbar() {
      return (
          <ThemedButton />
        </div>
    class ThemedButton extends React.Component {
      // 指定 contextType 读取当前的 theme context。
      // React 会往上找到最近的 theme Provider,然后使用它的值。
      // 在这个例子中,当前的 theme 值为 “dark”。
      static contextType = ThemeContext;
      render() {
        return <Button theme={this.context} />;
    function Button({ theme }) {
      return <button style={{ backgroundColor: theme }}>Test Button</button>;
    // 也可以使用 Consumer 获取 context
    // function ThemedButton () {
    //   return (
    //     <ThemeContext.Consumer>
    //       {value => <Button theme={value} />}
    //     </ThemeContext.Consumer>
    //   )
    // 或者使用 hooks,useContext()
    
    

    bindActionCreators 在 connect 时传 mapDispatchToProps 为对象时会用到,用于将对象中每个 key 对应的 value(action creator) 绑定一层 dispatch 调用。然后通过 props 传给组件,使组件中可以更方便的修改 redux 的 state,减少组件中写很多 dispatch(xxx)。

    action creator 后面会讲。

    用法: bindActionCreators(actionCreators, dispatch)

    bindActionCreators 把一个 value 为不同 action creator 的对象,转成拥有同名 key 的对象,同时使用 dispatch对每个 action creator 进行包装,以便可以直接调用它们。

    作用大概是下面这样的效果:

    add: () => ({ type: 'ADD' }), // 变成 add: () => dispatch({ type: 'ADD' })
  • actionCreators (Function or Object): 一个 action creator,或者一个 value 是 action creator 的对象。
  • dispatch (Function): 一个由 Store 实例提供的 dispatch 函数。
  • 8.4.1 Action Creator

    Action Creator 是指用来生成 action 的函数

    function ADD_TODO(text) {
    	return { type: 'ADD', text}
    dispatch(ADD_TODO('待办事项1')) // dispatch({ type: 'ADD', text: '待办事项1'})
    

    上例中 ADD_TODO 用来生成 Action { type: 'ADD', text}ADD_TODO 就是一个 action creator 函数

    Action 是一个信息的负载,而 action creator 是一个创建 action 的工厂。

    8.4.1 bindActionCreators 实现

    作用:在 mapDispatchToProps 为对象时,使对象 { add: () => ({type: 'ADD'})} 相当于 add = () => dispatch({type: 'ADD'})

    // bindActionCreator 会对 action creator 进行包装,加上 dispatch 调用
    function bindActionCreator(actionCreator, dispatch) {
      return (...args) => dispatch(actionCreator(...args));
    export default function bindActionCreators(actionCreators, dispatch) {
      if (typeof actionCreators === 'function') {
        return bindActionCreator(actionCreators, dispatch)
      const boundActionCreators = {};
      for (let key in actionCreators) {
        const actionCreator = actionCreators[key]
        if (typeof actionCreator === 'function') {
          boundActionCreators[key] = bindActionCreator(actionCreator, dispatch);
      return boundActionCreators;
    
    

    src/react-redux/index.js

    import Provider from './Provider';
    import connect from './connect';
    // 暴露 Provider,connect API
    export { Provider, connect };
    

    创建一个 Context,用于数据通信

    src/react-redux/Context.js

    import React from 'react';
    export const ReactReduxContext = React.createContext(null);
    export default ReactReduxContext;
    import React, { Component } from 'react';
    import { ReactReduxContext } from './Context';
    export default class Provider extends Component {
      render() {
        // 使用 Context.Provider 为组件提供 store
        return (
          <ReactReduxContext.Provider value={this.props.store}>
            {this.props.children}
          </ReactReduxContext.Provider>
    
    

    connect 接受 mapStateToProps 和 mapDispatchToProps 参数,返回一个高阶组件。

    “连接” 需要使用 store 的组件,把 store 中的值传递给组件