React更偏向于构建UI,虽然也能进行状态管理,但是面对大型复杂的项目,往往会面临跨组件通信,这时候就可以使用Redux。Redux是一种状态管理容器,他可以将组件用到的所有数据都保存在store中。

createStore

createStore(reducer, initialState, enhancer)用于创建一个store,第一个参数reducer用来修改store中的state,initialState为初始的state,enhancer是一个增强器,如果项目使用middleware的话,enhancer就是applyMiddleware的返回值,如果没有用middleware,则enhancer不传,

  1. export default function createStore(reducer, initialState, enhancer) {
  2. // ...
  3. if (typeof enhancer !== 'undefined') {
  4. if (typeof enhancer !== 'function') {
  5. throw new Error('Expected the enhancer to be a function.')
  6. }
  7. // 如果存在enhancer:enhancer(createStore)返回一个增强的(使用了middleware)createStore
  8. return enhancer(createStore)(reducer, initialState)
  9. }
  10. // ...
  11. function dispatch(action) {
  12. try {
  13. isDispatching = true
  14. currentState = currentReducer(currentState, action)
  15. } finally {
  16. isDispatching = false
  17. }
  18. return action
  19. }
  20. // State 是只读的,唯一改变 state 的方法就是触发 action,
  21. // reducer根据action返回一个新的state
  22. function getState() {
  23. return currentState
  24. }
  25. //如果不存在enhancer,则返回的store是这样的
  26. return {
  27. dispatch,
  28. subscribe,
  29. getState,
  30. replaceReducer,
  31. [$$observable]: observable
  32. }
  33. }

applyMiddleware

我们在上面说过了,applyMiddleware创建一个store的增强器,之所以增强,是因为dispatch被增强了,他是在原来的dispatch上附加了很多功能,具体是怎么实现的的,我们一起往下看

先给出中间件的写法

  1. const middleware1 = middlewareAPI => next => action => {
  2. console.log('m1 start')
  3. const result = next(action)
  4. console.log('m1 end')
  5. return result
  6. }
  7. const middleware2 = middlewareAPI => next => action => {
  8. console.log('m2 start')
  9. const result = next(action)
  10. console.log('m2 end')
  11. return result
  12. }
  1. export default function applyMiddleware(...middlewares) {
  2. return createStore => (...args) => {
  3. const store = createStore(...args)
  4. let dispatch = () => {
  5. throw new Error(
  6. 'Dispatching while constructing your middleware is not allowed. ' +
  7. 'Other middleware would not be applied to this dispatch.'
  8. )
  9. }
  10. const middlewareAPI = {
  11. getState: store.getState,
  12. dispatch: (...args) => dispatch(...args)
  13. }
  14. // 将上面两个middleware传入applyMiddleware,得到
  15. // chain = [dispatch1, dispatch2]
  16. const chain = middlewares.map(middleware => middleware(middlewareAPI))
  17. dispatch = compose(...chain)(store.dispatch)
  18. return {
  19. ...store,
  20. dispatch
  21. }
  22. }
  23. }
  24. // chain = [chain1, chain2]
  25. const chain1 = next => action => {
  26. console.log('m1 start')
  27. const result = next(action)
  28. console.log('m1 end')
  29. return result
  30. }
  31. const chain2 = next => action => {
  32. console.log('m2 start')
  33. const result = next(action)
  34. console.log('m2 end')
  35. return result
  36. }

compose

compose方法最核心的就是使用Array.prototype.reduce,具体使用方法可以查看MDN,经过reduce之后得到的结果

  1. export default function compose(...funcs) {
  2. if (funcs.length === 0) {
  3. return arg => arg
  4. }
  5. if (funcs.length === 1) {
  6. return funcs[0]
  7. }
  8. return funcs.reduce((a, b) => (...args) => a(b(...args)))
  9. }
  10. // compose(f, g, h) ====> (...args) => f(g(h(...args)))
  11. // composeResult = compose(...chain) = (...args) => chain1(chain2(...args))
  12. // dispatch = chain1(chain2(defaultDispatch))

新dispatch

我们用defaultDispatch表示最开始创建的dispatch(createStore中的dispatch),来看一下最后经过middleware包装过的dispatch长啥样

  1. const chain1 = next => action => {
  2. console.log('m1 start')
  3. const result = next(action)
  4. console.log('m1 end')
  5. return result
  6. }
  7. const chain2 = next => action => {
  8. console.log('m2 start')
  9. const result = next(action)
  10. console.log('m2 end')
  11. return result
  12. }
  13. dispatch = chain1(chain2(defaultDispatch))
  14. dispatch2 = action => {
  15. console.log('m2 start')
  16. const result = defaultDispatch(action)
  17. console.log('m2 end')
  18. return result
  19. }
  20. dispatch1 = action => {
  21. console.log('m1 start')
  22. const result = dispatch2(action)
  23. console.log('m1 end')
  24. return result
  25. }
  26. dispatch = action => {
  27. console.log('m1 start')
  28. // const result = dispatch2(action)
  29. // ----------------------------------
  30. console.log('m2 start')
  31. const result = defaultDispatch(action)
  32. console.log('m2 end')
  33. // ----------------------------------
  34. console.log('m1 end')
  35. return result
  36. }

这里也展示了middleware是洋葱模型,中间件的访问顺序是外 => 内 => 外,最内部其实是调用最原生的(createStore中定义的)dispatch,来调用reducer生成最新的state

另外可以看到middlewareAPI中的dispatch是

  1. const middlewareAPI = {
  2. getState: store.getState,
  3. dispatch: (...args) => dispatch(...args)
  4. }

而不是

  1. const middlewareAPI = {
  2. getState: store.getState,
  3. dispatch: dispatch
  4. }

是因为这里使用的是闭包,这样可以将经过中间件生成新的dispatch传给每一个中间件。