redux源码阅读之compose,applyMiddleware

我的观点是,看别人的源码,不追求一定要能原样造轮子,单纯就是学习知识,对于程序员的提高就足够了。在阅读redux的compose源码之前,我们先学一些前置的知识。

redux源码阅读之compose,applyMiddleware

看别人的源码就是学习知识。我们先学一些东西。

rest参数

形式为...变量名,用于获取函数的多余参数 ,该变量将多余的参数放入数组中, 只能是参数的最后一个。

function rest(...args){
  console.log(args);
};
rest('arg1','arg2')

那么args变量就是一个数组['arg1','arg2']。

扩展运算符

扩展运算符也是三个点。好比rest参数的逆运算,将一个数组转为用逗号分隔的参数序列

arr.reduce(callback[initialValue])

个人感觉reduce用法非常魔幻,要搞明白它之后,才能明白compose。
MDN的reduce文档:
https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Array/Reduce

使用

const array1 = [1, 2, 3, 4];
const reducer = (accumulator, currentValue) => accumulator + currentValue;

// 1 + 2 + 3 + 4
console.log(array1.reduce(reducer));
// 10

// 5 + 1 + 2 + 3 + 4
console.log(array1.reduce(reducer, 5));
// 15

array1.reduce((accumulator, currentValue) => accumulator + currentValue,5)

根据MDN的定义,
数组的reduce() 方法对数组中的每个元素执行一个由您提供的reducer函数(升序执行),将其结果汇总为单个返回值。

arr.reduce(callback[initialValue])

实际就是arr.reduce(callback, initialValue);

callback 函数接收4个参数:

Accumulator (acc) (累加器)
Current Value (cur) (当前值)
Current Index (idx) (当前索引)
Source Array (src) (源数组)

initialValue 可选

作为第一次调用 callback函数时的第一个参数的值。 如果没有提供初始值,则将使用数组中的第一个元素。 不传这个初始值initialVale,且在空数组上调用 reduce 将报 Uncaught TypeError: Reduce of empty array with no initial value的错误。

pip

研究compose之前,我们再来看MDN上reduce的应用,一个功能型函数管道。https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Array/Reduce#功能型函数管道

类似的我们来写一个。

const pipe = (...fns) => 
  (args) => fns.reduce(
    (args, fn) => fn(args)
  ,args)

吐槽一下,就是这种es6写法,完全不写return,很多es5过来的人不习惯,所以很多源码很难看清楚。我们改写一下,传入参数字符串‘test’,然后执行这个函数。就能明白pip函数的作用就是顺序执行了addFont,toUpper。

args就是'test',...fns就是rest参数,pip函数里的fns参数就是数组[addfont,toUpper]。


const pipe = (...fns) => {
  return (args) => {
    return fns.reduce((args, fn) => {
      return fn(args)
    }, args)
  }
}

const toUpper = (value) => {
  return value.toUpperCase();
}

const addFont = (value) => {
  return 'hello plus!' + value;
}

console.log(pipe(addFont,toUpper)('test'));
// HELLO PLUS!TEST

pipe(addFont,toUpper)之后,返回一个函数.
pipe(addFont,toUpper)('test')形如

(args) => {
    return fns.reduce((args, fn) => {
      return fn(args)
    }, args)
  }('test');

再执行就是

fns.reduce(('test',fn)=>{
    return fn('test');
},'test')

这样子看代码,这里的reduce应该似乎比较好懂的。

compose

compose源码

/**
 * Composes single-argument functions from right to left. The rightmost
 * function can take multiple arguments as it provides the signature for
 * the resulting composite function.
 *
 * @param {...Function} funcs The functions to compose.
 * @returns {Function} A function obtained by composing the argument functions
 * from right to left. For example, compose(f, g, h) is identical to doing
 * (...args) => f(g(h(...args))).
 */

export default function compose(...funcs) {
  if (funcs.length === 0) {
    return arg => arg
  }

  if (funcs.length === 1) {
    return funcs[0]
  }

  return funcs.reduce((a, b) => (...args) => a(b(...args)))
}

跟pip函数对比

console.log(pipe(addFont,toUpper)('test'));
// HELLO PLUS!TEST

console.log(compose(addFont,toUpper)('test'));
// hello plus!TEST

compose正好相反,后添加的方法先执行。然后写的也很玄学让人费解。让我们改成pip函数一样,也是同样的效果,感觉这样子就跟上面的pip一样好理解一些了呢。

export default function compose(...funcs) {
  if (funcs.length === 0) {
    return arg => arg
  }

  if (funcs.length === 1) {
    return funcs[0]
  }

  return (...funcs) => (...args) => funcs.reduce((a, b)  => a(b(...args)))
}
console.log(compose(addFont,toUpper)('test'));
// hello plus!TEST

applyMiddleware

我们再来看redux引入中间件的用法,

const store = createStore(
  reducer,
  applyMiddleware(...middleware)
);

以及applyMiddleware方法源码

export default function applyMiddleware(...middlewares) {
  return createStore => (...args) => {
    const store = createStore(...args)
    let dispatch = () => {
      throw new Error(
        `Dispatching while constructing your middleware is not allowed. ` +
          `Other middleware would not be applied to this dispatch.`
      )
    }

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

    return {
      ...store,
      dispatch
    }
  }
}

实际上所谓redux的中间件就是改写了store.dipatch方法。比如我们自己实现一个实现一个打印action的type的redux中间件,我们就可以这样写。

./redux-middlewares/log.js

const logger = ()=>{
    return ({dispatch,getState}) => next => action =>{
        console.log('dispatching', action);
        return next(action);
    }
};

export default logger;

跟别的redux中间件一起引入

import createSaga from 'redux-saga';
import createLog from './redux-middlewares/log';
const saga = createSaga(); 
const log = createLog();
const middlewares = [saga,log]
const store = createStore(
  rootReducer,
  applyMiddleware(...middlewares)
);
store.dispatch({type:'myAction'});

// 打印 dispatching {type: "myAction"}

这里
redux的添加中间件
applyMiddleware(...middleware) 就等于这里的
dispatch = compose(...chain)(store.dispatch).
跟我们实行
compose(addFont,toUpper)('test')是一样的效果。

参考文章:

https://www.cnblogs.com/cloud-/p/7282188.html

posted @ 2018-12-24 02:07  从过去穿越到现在  阅读(264)  评论(0编辑  收藏  举报