美文网首页
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

    applyMiddleware.ts 函数重载声明 首先是对applyMiddleware函数的重载,重载了七个函...

  • redux 源码阅读

    redux 源码阅读 首先从 redux 的官方示例来看 redux 的作用 这样简单一看的话, redux 感觉...

  • redux源码阅读

    Redux 是可预测的状态管理框架,它很好的解决多交互,多数据源的诉求。 三大原则: 单一数据源:整个应用的sta...

  • redux源码阅读——applyMiddleware

    redux源码——applyMiddleware 相关源码展示 applyMiddleware源码 compose...

  • redux源码阅读记录

    标签(空格分隔): redux javascirpt createStore 导语: 最近在看 redux,也看了...

  • Redux源码阅读_2

    compose.ts 从右到左来组合多个函数,是reduce函数的一个应用实现。 首先仍然是重载了多个参数的函数声...

  • Redux源码阅读_3

    combineReducers.ts 函数重载声明 首先是对combineReducers函数的重载,重载了三个函...

  • redux源码阅读笔记

    砖头搬完了,找点乐趣,记得之前看vuex的源码.. 挺有意思,没有记录..现在忘光光.. 这一年一直写reac...

  • redux源码分析

    这篇文章会很长。redux源码读起来并不困难。读懂redux的源码有利于更好的使用redux。但是react-re...

  • 重读redux源码

    之前5.1假期的时候,有去读了下redux的源码,当时感触很深,今天重新梳理遍redux的源码:从源码中的src文...

网友评论

      本文标题:Redux源码阅读_4

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