如何理解redux中间件(一)

“中间件”这个词听起来很恐怖,但它实际一点都不难。想更好的了解中间件的方法就是看一下那些已经实现了的中间件是怎么工作的,然后尝试自己写一个。函数嵌套写法看起来很恐怖,但是大多数你能找到的中间件,代码都不超过十行,但是它们的强大来自于它们的可嵌套组合性。

这段话我们提取关键词后得出:我们需要了解什么是函数嵌套,中间件就是变态的函数嵌套罢了,函数嵌套大约就是函数式编程的概念,而使用函数式编程必须知道的就是柯里化。

需要知道的概念

  • 函数式编程
  • 柯里化
  • middleware

这里从middleware开始说起(函数式编程,柯里化概念请自行去理解),就拿异步actionredux-thunk( 简写版)来说

export default const thunkMiddleware = 
  ({ dispatch, getState }) =>
        next => 
             action => 
                   typeof action === ‘function’ ? 
                     action(dispatch, getState) : 
                     next(action);

Redux中文文档如是解释:

...middlewares (arguments): 遵循 Redux middleware API 的函数。每个 middleware 接受 Store
dispatch
getState
函数作为命名参数,并返回一个函数。该函数会被传入 被称为 next的下一个 middleware 的 dispatch 方法,并返回一个接收 action 的新函数,这个函数可以直接调用 next(action),或者在其他需要的时刻调用,甚至根本不去调用它。调用链中最后一个 middleware 会接受真实的 store 的 dispatch 方法作为 next 参数,并借此结束调用链。所以,middleware 的函数签名是 ({ getState,dispatch }) => next => action

其中storedispatchgetState方法是在createStore时通过applyMiddleware注入的。

next 方法用于维护中间件调用链和dispatch,它返回一个接受 action 对象的柯里化函数,接受的 action 对象可以在中间件中被修改,再传递给下一个被调用的中间件,最终dispatch会使用中间件修改后的action来执行。

如果得到的action是个函数,就用 dispatchgetState 当作参数来调用它,否则就直接分派给store。

再来看看我们异步的action写法:

export const test = (params)=> {
      return (dispatch, getState)=> {
            // do something
            dispatch(otherAction)
      }
}

到这一步,我们对于为什么要使用redux-thunk中间件有了合理的解释,还是不理解可以参考:redux-tutorial-cn

创建store的过程

理解middleware我们从创建store的源头开始,其中createStore最常见的写法:

const enhancer = compose(
    // 应用中间件到store中
    applyMiddleware(reduxPromiseMiddleware(), funActionThunk, thunk)
)
export default const store = createStore(reducer, initialState, enhancer)

从上得知,入口从createStore方法开始,所以来先瞅一眼createStore.js源码

export default function createStore(reducer, preloadedState, enhancer) {
  if (typeof preloadedState === 'function' && typeof enhancer === 'undefined') {
    enhancer = preloadedState
    preloadedState = undefined
  }

  if (typeof enhancer !== 'undefined') {    
    if (typeof enhancer !== 'function') {
      throw new Error('Expected the enhancer to be a function.')
    }
   // 根据我们的传参该函数会在此处返回,将其本身作为参数传递给enhancer函数,并再次调用传递剩余参数
    return enhancer(createStore)(reducer, preloadedState)
  }

...省略

从上面代码的注释处可以看出,调用了enhancer函数,而enhancer函数由applyMiddleware后得到,所以再来看看applyMiddleware.js的源码:

export default function applyMiddleware(...middlewares) {
  return (createStore) => (reducer, preloadedState, enhancer) => {
    var store = createStore(reducer, preloadedState, enhancer)
    var dispatch = store.dispatch
    var chain = []

    var middlewareAPI = {
      getState: store.getState,
      dispatch: (action) => dispatch(action)
    }
    chain = middlewares.map(middleware => middleware(middlewareAPI))
    dispatch = compose(...chain)(store.dispatch)

    return {
      ...store,
      dispatch
    }
  }
}

此处看到它返回一个函数,而这个函数刚好接受enhancer函数的参数createStore,而这个createStore就是redux封装的方法,接着返回一个函数,该函数的三个参数和创建store时的createStore方法传的参数一致,只不过在这里enhancer === ‘undefined’,此时正好返回了初始化后的store:

var store = createStore(reducer, preloadedState, enhancer)

这个store包含4个方法,见createStore源码:

return {
    dispatch,
    subscribe,
    getState,
    replaceReducer,
    [$$observable]: observable
  }

将dispatch用另外变量保存的目的就是为了接下来将dispatch getState注入给所有的middleware

chain = middlewares.map(middleware => middleware(middlewareAPI))

然后将多个middleware函数经过compose处理后得到一个函数,这一步形象点说就是将最初原始的dispatch经过一层一层middleware中间件“加工后”重新包装并返回生成终极dispatch和初始化后的store

进一步

理解完了源码的流程我们最后再来回顾中间件redux-thunk源码,以此来理解,中间件是如何包装dispatch的,不然我们讨论这些有何意义?

export default const thunkMiddleware = 
  ({ dispatch, getState }) =>
        next => 
             action => 
                   typeof action === ‘function’ ? 
                     action(dispatch, getState) : 
                     next(action);

现在我们来看源码是不是很清晰?中间件只能接受这2个参数{ dispatch, getState },因为它就是middlewareAPI

  • next是什么
    中间件返回一个函数,参数是next,然后再联系applyMiddleware源码来看
dispatch = compose(...chain)(store.dispatch)

到这里我们不得不来看下compose的作用是什么了

compose 做的只是让你不使用深度右括号的情况下来写深度嵌套的函数 (这里不展开,可自行详细了解).

我们可以看到最右侧的middleware传入的第一个参数就是store.dispatch,然后以返回的结果作为参数继续传递给下一个中间件,最终返回dispatch。所以,这里的next就是最原始的dispatch,或者经过前面的middleware处理过后的dispatch

  • action是什么?
    上面我们可以看到,最终返回的这个终极dispatch成了我们在action中调用的dispatch,让我们回归到最初的调用:
export const test = (params)=> {
      return (dispatch, getState)=> {  // 这个dispatch是终极dispatch吗?答案是
            // do something
            dispatch(otherAction). // 这个dispatch是终极版dispatch吗?答案不是,有可能是最原始的,也有可能是前一个middleware处理过后的
      }
}

这里可以看到,这个action返回一个函数,这个函数接受的参数就是我们创建store时调用createStore方法返回的参数(这里只是选取了2个)。要说明的是,这里的dispatch就是终极版的dispatch,中间件里的action是我们在定义Actionreturn的方法或者dispatchotherAction,所以我们看到例子中的test方法(异步action)返回的是一个函数,redux-thunk源码中判断action是一个函数后就返回了一个函数,这个函数的参数就是return (dispatch, getState)=> {}里的参数,给后面触发action用。

  • middleware是如何处理“加工”dispatch的?
    因为我们前面说过:
dispatch = compose(...chain)(store.dispatch)

middleware chain经过compose后从右到左执行一层一层处理dispatch,其实这个表述不准,或者我可以这样理解:中间件就是一个闭包?在拿到dispatch后保存在这里,当action被触发后,经过异步的操作后,用这个dispatch触发了需要异步处理结束后再触发的action。(此处理解可能不妥)

这样一来,也方便我们理解为什么要使用redux-thunk中间件来实现异步action了,因为你可以在dispatch之前做其它操作再触发,感觉就像是闭包的概念,dispatch作为闭包中的变量方法,闭包中含有其它异步操作,直到异步操作完成之后才被触发调用。(这里的闭包概念引用出处:前端基础进阶(六):在chrome开发者工具中观察函数调用栈、作用域链与闭包

最后,希望也请各位看官大爷有错必究,不要只顾着喷小弟理解不到位的地方,[跪谢]。

如何理解redux中间件(二) 敬请期待

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,088评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,715评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,361评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,099评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 60,987评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,063评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,486评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,175评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,440评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,518评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,305评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,190评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,550评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,880评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,152评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,451评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,637评论 2 335

推荐阅读更多精彩内容