美文网首页
Redux源码阅读_4

Redux源码阅读_4

作者: 晴窗细语 | 来源:发表于2020-08-10 19:25 被阅读0次

    applyMiddleware.ts

    函数重载声明

    首先是对applyMiddleware函数的重载,重载了七个函数,主要是传入参数个数的区别。

    export default function applyMiddleware(): StoreEnhancer
    export default function applyMiddleware<Ext1, S>(
      middleware1: Middleware<Ext1, S, any>
    ): StoreEnhancer<{ dispatch: Ext1 }>
    export default function applyMiddleware<Ext1, Ext2, S>(
      middleware1: Middleware<Ext1, S, any>,
      middleware2: Middleware<Ext2, S, any>
    ): StoreEnhancer<{ dispatch: Ext1 & Ext2 }>
    export default function applyMiddleware<Ext1, Ext2, Ext3, S>(
      middleware1: Middleware<Ext1, S, any>,
      middleware2: Middleware<Ext2, S, any>,
      middleware3: Middleware<Ext3, S, any>
    ): StoreEnhancer<{ dispatch: Ext1 & Ext2 & Ext3 }>
    export default function applyMiddleware<Ext1, Ext2, Ext3, Ext4, S>(
      middleware1: Middleware<Ext1, S, any>,
      middleware2: Middleware<Ext2, S, any>,
      middleware3: Middleware<Ext3, S, any>,
      middleware4: Middleware<Ext4, S, any>
    ): StoreEnhancer<{ dispatch: Ext1 & Ext2 & Ext3 & Ext4 }>
    export default function applyMiddleware<Ext1, Ext2, Ext3, Ext4, Ext5, S>(
      middleware1: Middleware<Ext1, S, any>,
      middleware2: Middleware<Ext2, S, any>,
      middleware3: Middleware<Ext3, S, any>,
      middleware4: Middleware<Ext4, S, any>,
      middleware5: Middleware<Ext5, S, any>
    ): StoreEnhancer<{ dispatch: Ext1 & Ext2 & Ext3 & Ext4 & Ext5 }>
    export default function applyMiddleware<Ext, S = any>(
      ...middlewares: Middleware<any, S, any>[]
    ): StoreEnhancer<{ dispatch: Ext }>
    
    函数实现
    export default function applyMiddleware(
      ...middlewares: Middleware[]
    ): StoreEnhancer<any> {
      return (createStore: StoreEnhancerStoreCreator) => <S, A extends AnyAction>(
        reducer: Reducer<S, A>,
        preloadedState?: PreloadedState<S>
      ) => {
        const store = createStore(reducer, preloadedState)
        let dispatch: Dispatch = () => {
          throw new Error(
            'Dispatching while constructing your middleware is not allowed. ' +
              'Other middleware would not be applied to this dispatch.'
          )
        }
    
        const middlewareAPI: MiddlewareAPI = {
          getState: store.getState,
          dispatch: (action, ...args) => dispatch(action, ...args)
        }
        const chain = middlewares.map(middleware => middleware(middlewareAPI))
        dispatch = compose<typeof dispatch>(...chain)(store.dispatch)
    
        return {
          ...store,
          dispatch
        }
      }
    }
    

    可以看到applyMiddleware返回一个对象,对象内容为{...store, dispatch}。

    首先看看定义的两个接口 MiddlewareAPI与Middleware:

    export interface MiddlewareAPI<D extends Dispatch = Dispatch, S = any> {
      dispatch: D
      getState(): S
    }
    
    export interface Middleware<
      _DispatchExt = {}, // TODO: remove unused component (breaking change)
      S = any,
      D extends Dispatch = Dispatch
    > {
      (api: MiddlewareAPI<D, S>): (
        next: D
      ) => (action: D extends Dispatch<infer A> ? A : never) => any
    }
    

    可以看到,MiddlewareAPI是一个包含dispath与state属性的对象,而Middleware是一个高阶函数,将Middleware写成函数形式,其结构大致为:

    function Middleware(api: ...) {
        return (next: ...) => (action: ...) {
            ...
        }
    }
    

    若在上面的结构上做出如下类型申明:

    type Next<...> = (next: ...) => (action: ...) => any
    

    则可以看做Middleware的返回值类型是个Next。

    回到applyMiddleware函数代码中,带入参数看看函数内部运作。

    若传入参数 middlewares = [middleware1, middleware2, middleware3]

    则 chain = [next1, next2, next3]

    由前文提到 compose实现( Redux源码阅读_2

    dispatch = compose<typeof dispatch>(...chain)(store.dispatch)
                  =next1(next2(next3(store.dispatch)))

    就是个逐层进行函数运算的操作。

    后记

    redux源码到这里就告一段落(还有个bindActionCreator函数实现没整理,但是平时没怎么用到就不弄了),回到最开始的问题,基本都是react-redux的内容,这个之后再去看……

    只有一个可以回到的问题,action相关……

    看看redux关于action的声明

    export interface Action<T = any> {
      type: T
    }
    

    就是一个包含type属性的对象…… dispatch action的操作在前面关于dispatch函数的分析中也写到了……

    文档上写到的基本也就是源码的所有内容,不过只看文档还是有点不知所云,结合源码还是更能理解一点。(给自己找点借口……)

    相关文章

      网友评论

          本文标题:Redux源码阅读_4

          本文链接:https://www.haomeiwen.com/subject/yaojdktx.html