成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Redux系列源碼解讀

Scliang / 1019人閱讀

摘要:源碼解讀的方法就是給我們提供了靈活的創(chuàng)建符合標(biāo)準(zhǔn)的的方法。用于分解樹(shù),每一個(gè)對(duì)應(yīng)的一個(gè)對(duì)應(yīng)的子。這樣在將傳遞給時(shí)利于分解。源碼實(shí)現(xiàn)了這種將函數(shù)數(shù)組,通過(guò)的方法,實(shí)現(xiàn)層層嵌套的執(zhí)行,達(dá)到中間件的實(shí)現(xiàn)。

Redux 源碼解讀 1.redux-action createAction

redux-action的createAction方法就是給我們提供了靈活的創(chuàng)建符合FSA標(biāo)準(zhǔn)的action的方法。

exports.default = createAction;
 
/**
  返回創(chuàng)建action的函數(shù)
*/
function createAction(type, payloadCreator, metaCreator) {
  var finalPayloadCreator = typeof payloadCreator === "function" ? payloadCreator : _identity2.default;
  /**
    返回的函數(shù)
  */
  var actionHandler = function actionHandler() {
    var hasError = (arguments.length <= 0 ? undefined : arguments[0]) instanceof Error;
  /**
    返回的action
  */
    var action = {
      type: type
    };
    //根據(jù)傳入的參數(shù),執(zhí)行payloadCreator獲取payload
    var payload = hasError ? arguments.length <= 0 ? undefined : arguments[0] : finalPayloadCreator.apply(undefined, arguments);
    if (!(payload === null || payload === undefined)) {
      action.payload = payload;
    }
 
    if (hasError) {
      // Handle FSA errors where the payload is an Error object. Set error.
      action.error = true;
    }
  //根據(jù)傳入的參數(shù),執(zhí)行metaCreator獲取payload
    if (typeof metaCreator === "function") {
      action.meta = metaCreator.apply(undefined, arguments);
    }
    //可以看到  payloadCreator和metaCreator的參數(shù)都是用的傳給actionHandler的參數(shù)
 
    return action;
  };
 
  actionHandler.toString = function () {
    return type.toString();
  };
 
  return actionHandler;
}
2.redux combineReducer

redux的combineReducer方法 用于將多個(gè)reducer,合并成一個(gè)大的reducer函數(shù),傳給store。

用于分解state樹(shù),每一個(gè)reducer對(duì)應(yīng)state的一個(gè)key對(duì)應(yīng)的子state。比如poi的reducer對(duì)應(yīng)的就是state[poi]。這樣在將state傳遞給props時(shí)利于分解。

reducer以對(duì)象的形式傳入,finalReducers 存放最終的reducer,finalReducerKeys存放reducer的key
最終返回  combination函數(shù) reducer類型的函數(shù),接受state和action 返回state
 
state的形式是一個(gè)大對(duì)象下面每一個(gè)reducer對(duì)應(yīng)一個(gè)子state。
 
觸發(fā)一個(gè)action,會(huì)遍歷所有的reducer,
將該reducer的舊state和action傳入,然后根據(jù)返回的新的state對(duì)象是否改變,來(lái)決定
最終的返回的state是否改變。
這里需要注意:由于state都是引用類型,這里比較是值比較
hasChanged = hasChanged || nextStateForKey !== previousStateForKey
 
所以如果我們想要改變?nèi)值膕tate,需要在reducer中返回新的對(duì)象,而不是原來(lái)的state對(duì)象,
如果返回原來(lái)的對(duì)象,即使對(duì)象里的值改變了,也不會(huì)引起全局state的改變。
 */
export default function combineReducers(reducers) {
  var reducerKeys = Object.keys(reducers)
  var finalReducers = {}
  for (var i = 0; i < reducerKeys.length; i++) {
    var key = reducerKeys[i]
 
    if (process.env.NODE_ENV !== "production") {
      if (typeof reducers[key] === "undefined") {
        warning(`No reducer provided for key "${key}"`)
      }
    }
 
    if (typeof reducers[key] === "function") {
      finalReducers[key] = reducers[key]
    }
  }
  var finalReducerKeys = Object.keys(finalReducers)
 
  return function combination(state = {}, action) {
    /**
     校驗(yàn)語(yǔ)法錯(cuò)誤,reducer返回的state不能是undefined
    */
    if (sanityError) {
      throw sanityError
    }
 
    var hasChanged = false
    var nextState = {}
    for (var i = 0; i < finalReducerKeys.length; i++) {
      var key = finalReducerKeys[i]
      var reducer = finalReducers[key]
      var previousStateForKey = state[key]
      var nextStateForKey = reducer(previousStateForKey, action)
      if (typeof nextStateForKey === "undefined") {
        var errorMessage = getUndefinedStateErrorMessage(key, action)
        throw new Error(errorMessage)
      }
      nextState[key] = nextStateForKey
      /**
        所以如果我們想要改變?nèi)值膕tate,需要在reducer中返回新的對(duì)象,而不是原來(lái)的state對(duì)象,
        如果返回原來(lái)的對(duì)象,即使對(duì)象里的值改變了,也不會(huì)引起全局state的改變。
      */
      hasChanged = hasChanged || nextStateForKey !== previousStateForKey
    }
    return hasChanged ? nextState : state
  }
}
3 redux applyMiddleware

應(yīng)用中間件的目的是包裝dispatch,在action傳遞給dispatch執(zhí)行之前,需要經(jīng)過(guò)中間件的層層處理,進(jìn)行一些業(yè)務(wù)上的處理,決定action的走向。

源碼實(shí)現(xiàn)了這種將函數(shù)數(shù)組,通過(guò)reducerRight的方法,實(shí)現(xiàn)層層嵌套的執(zhí)行,達(dá)到中間件的實(shí)現(xiàn)。

/**
  顯示執(zhí)行中間件,得到中間件的返回函數(shù)數(shù)組chain,然后利用compose方法,生成嵌套的執(zhí)行chain
  方法的包裝dispatch函數(shù),
  中間件的形式是
  (getState, dispatch)=> next => action => {
     next(action);
  }
 */
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)
    /**
    store.dispatch 就是第一個(gè)next  是last ware的next
    (...args) => {
      return ware0(ware1(ware2(last(...args))))
    }
    dispatch = ware0(ware1(ware2(last(...args))))
    所以中間件中next傳入后返回的函數(shù)就是我們需要的函數(shù)形式,
    例如dispatch 需要的函數(shù)形式是 傳一個(gè)action
    */
    return {
      ...store,
      dispatch
    }
  }
}
 
/**
reduceRight是數(shù)組的從右至左執(zhí)行,
初始的參數(shù)是最后一個(gè)函數(shù)接受dispatch,
的到的一個(gè)action=>{
    dispatch(action);
}
形式的函數(shù),作為參數(shù)composed
f的形式是
next=>action=>{
}
最終形成的就是
(...args) => {
    return funcs0(funcs1(funcs2(last(...args))))
}
*/
export default function compose(...funcs) {
  if (funcs.length === 0) {
    return arg => arg
  }
 
  if (funcs.length === 1) {
    return funcs[0]
  }
 
  const last = funcs[funcs.length - 1]
  const rest = funcs.slice(0, -1)
  return (...args) => rest.reduceRight((composed, f) => f(composed), last(...args))
}

中間件執(zhí)行過(guò)程模擬

中間件原理
*/
function func1 (next) {
    console.log("func1 return");
    return function (action) {
        console.log("func1start");
        next(action);
        console.log("func1end");
    }
     
}
function func2 (next) {
    console.log("func2 return");
    return function (action) {
        console.log("func2start");
        next(action);
        console.log("func2end");
    }
}
function func3 (next) {
    console.log("func3 return");
    return function (action) {
        console.log("func3start");
        next(action);
        console.log("func3end");
    }
}
 
function dispatch(action) {
    console.log(action);
}
 
function afterCompose (args) {
    return func1(func2(func3(args)));
}
 
var newdispatch = afterCompose(dispatch);
newdispatch("action");
/**
    執(zhí)行順序
    func3 return
    func2 return
    func1 return
    func1start
    func2start
    func3start
    action
    func3end
    func2end
    func1end
*/
4 redux createStore

應(yīng)用場(chǎng)景參見(jiàn)中間件的應(yīng)用代碼與applyMiddleware源碼,是redux提供創(chuàng)建store的方法。

import isPlainObject from "lodash/isPlainObject"
import $$observable from "symbol-observable"
 
export var ActionTypes = {
  INIT: "@@redux/INIT"
}
 
export default function createStore(reducer, preloadedState, enhancer) {
  var currentReducer = reducer
  var currentState = preloadedState
  var currentListeners = []
  var nextListeners = currentListeners
  var isDispatching = false
 
  function ensureCanMutateNextListeners() {
    if (nextListeners === currentListeners) {
      nextListeners = currentListeners.slice()
    }
  }
 
  function getState() {
    return currentState
  }
 
   /**
    訂閱監(jiān)聽(tīng)
   */
  function subscribe(listener) {
    if (typeof listener !== "function") {
      throw new Error("Expected listener to be a function.")
    }
 
    var isSubscribed = true
 
    ensureCanMutateNextListeners()
    nextListeners.push(listener)
 
    return function unsubscribe() {
      if (!isSubscribed) {
        return
      }
 
      isSubscribed = false
 
      ensureCanMutateNextListeners()
      var index = nextListeners.indexOf(listener)
      nextListeners.splice(index, 1)
    }
  }
 
   /**
    執(zhí)行reducer,獲取state,執(zhí)行l(wèi)istener
   */
  function dispatch(action) {
    try {
      isDispatching = true
      currentState = currentReducer(currentState, action)
    } finally {
      isDispatching = false
    }
 
    var listeners = currentListeners = nextListeners
    for (var i = 0; i < listeners.length; i++) {
      listeners[i]()
    }
    return action
  }
 
   /**
    替換reducer
   */
  function replaceReducer(nextReducer) {
    if (typeof nextReducer !== "function") {
      throw new Error("Expected the nextReducer to be a function.")
    }
    currentReducer = nextReducer
    dispatch({ type: ActionTypes.INIT })
  }
 
 
  /**
    store創(chuàng)建的時(shí)候,獲取初始的sate樹(shù)
  */
  dispatch({ type: ActionTypes.INIT })
 
  return {
    dispatch,
    subscribe,
    getState,
    replaceReducer
  }
}
5. react-redux Provider

redux和react的結(jié)合,Provider作為根組件,將store的state放在context中供子組件使用。

import { Component, PropTypes, Children } from "react"
import storeShape from "../utils/storeShape"
import warning from "../utils/warning"
 
 
export default class Provider extends Component {
  //把 store 放在context里面,給子元素用
  getChildContext() {
    return { store: this.store }
  }
 
  constructor(props, context) {
    super(props, context)
    this.store = props.store
  }
 
  render() {
    const { children } = this.props
    //渲染唯一的子元素
    return Children.only(children)
  }
}
 
Provider.propTypes = {
  store: storeShape.isRequired,
  children: PropTypes.element.isRequired
}
Provider.childContextTypes = {
  store: storeShape.isRequired
}
6. react-redux connect

connect方法,將React的組件進(jìn)行包裝,包裝的目的如下:

能夠?qū)tore中指定的state,傳遞給組件當(dāng)props

能夠監(jiān)聽(tīng)store中state的變化

能夠?qū)ction傳遞給view

const defaultMapStateToProps = state => ({}) // eslint-disable-line no-unused-vars
const defaultMapDispatchToProps = dispatch => ({ dispatch })
const defaultMergeProps = (stateProps, dispatchProps, parentProps) => ({
  ...parentProps,
  ...stateProps,
  ...dispatchProps
})
 
 
export default function connect(mapStateToProps, mapDispatchToProps, mergeProps, options = {}) {
 
  //返回包裝組件的函數(shù)
  return function wrapWithConnect(WrappedComponent) {
 
    class Connect extends Component {
      shouldComponentUpdate() {
        return !pure || this.haveOwnPropsChanged || this.hasStoreStateChanged
      }
 
      constructor(props, context) {
        super(props, context)
        this.version = version
        this.store = props.store || context.store
 
       
        const storeState = this.store.getState()
        this.state = { storeState }
        this.clearCache()
      }
 
      isSubscribed() {
        return typeof this.unsubscribe === "function"
      }
 
      trySubscribe() {
        if (shouldSubscribe && !this.unsubscribe) {
          //訂閱store的state變化
          this.unsubscribe = this.store.subscribe(this.handleChange.bind(this))
          this.handleChange()
        }
      }
 
      tryUnsubscribe() {
        if (this.unsubscribe) {
          this.unsubscribe()
          this.unsubscribe = null
        }
      }
 
      componentDidMount() {
        //訂閱store的state變化
        this.trySubscribe()
      }
 
      componentWillReceiveProps(nextProps) {
        if (!pure || !shallowEqual(nextProps, this.props)) {
          this.haveOwnPropsChanged = true
        }
      }
 
      componentWillUnmount() {
        this.tryUnsubscribe()
        this.clearCache()
      }
 
      //訂閱變化
      handleChange() {
        if (!this.unsubscribe) {
          return
        }
 
        const storeState = this.store.getState()
        const prevStoreState = this.state.storeState
        if (pure && prevStoreState === storeState) {
          return
        }
 
        if (pure && !this.doStatePropsDependOnOwnProps) {
          const haveStatePropsChanged = tryCatch(this.updateStatePropsIfNeeded, this)
          if (!haveStatePropsChanged) {
            return
          }
          if (haveStatePropsChanged === errorObject) {
            this.statePropsPrecalculationError = errorObject.value
          }
          this.haveStatePropsBeenPrecalculated = true
        }
 
        this.hasStoreStateChanged = true
        //如果有變化 setState,觸發(fā)render
        this.setState({ storeState })
      }
 
      render() {
        const {
          haveOwnPropsChanged,
          hasStoreStateChanged,
          haveStatePropsBeenPrecalculated,
          statePropsPrecalculationError,
          renderedElement
        } = this
 
        
        //最終渲染組件,將合并屬性傳遞給WrappedComponent
        if (withRef) {
          this.renderedElement = createElement(WrappedComponent, {
            ...this.mergedProps,
            ref: "wrappedInstance"
          })
        } else {
          this.renderedElement = createElement(WrappedComponent,
            this.mergedProps
          )
        }
 
        return this.renderedElement
      }
    }
 
    Connect.displayName = connectDisplayName
    Connect.WrappedComponent = WrappedComponent
    Connect.contextTypes = {
      store: storeShape
    }
    Connect.propTypes = {
      store: storeShape
    }
 
    //把WrappedComponent的非靜態(tài)react屬性 復(fù)制到Connect,最終返回Connect
    return hoistStatics(Connect, WrappedComponent)
  }
}
7. redux bindActionCreators

使用實(shí)例參見(jiàn) react-redux connect 的使用實(shí)例

function bindActionCreator(actionCreator, dispatch) {
  return (...args) => dispatch(actionCreator(...args))
}
 
 
  將actionCreators綁定上dispatch,key還是actionCreators的key,但是多做了一層dispatch
 */
export default function bindActionCreators(actionCreators, dispatch) {
  if (typeof actionCreators === "function") {
    return bindActionCreator(actionCreators, dispatch)
  }
 
  if (typeof actionCreators !== "object" || actionCreators === null) {
    throw new Error(
      `bindActionCreators expected an object or a function, instead received ${actionCreators === null ? "null" : typeof actionCreators}. ` +
      `Did you write "import ActionCreators from" instead of "import * as ActionCreators from"?`
    )
  }
 
  var keys = Object.keys(actionCreators)
  var boundActionCreators = {}
  for (var i = 0; i < keys.length; i++) {
    var key = keys[i]
    var actionCreator = actionCreators[key]
    if (typeof actionCreator === "function") {
      boundActionCreators[key] = bindActionCreator(actionCreator, dispatch)
    }
  }
  return boundActionCreators
}

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/80813.html

相關(guān)文章

  • dva系列源碼解讀

    摘要:介紹概述本次對(duì)源碼的解讀除了傳統(tǒng)的從入手外還將引入帶入問(wèn)題讀源碼的理念,因?yàn)橹挥羞@樣當(dāng)讀完源碼之后才會(huì)有切身的收獲。 介紹 概述 本次對(duì) dva 源碼的解讀除了傳統(tǒng)的從 api 入手外還將引入帶入問(wèn)題讀源碼的理念,因?yàn)橹挥羞@樣當(dāng)讀完源碼之后才會(huì)有切身的收獲。另外除了 dva 的源碼外還會(huì)解讀一些常用的 dva 插件的源碼。 電子書(shū) https://dva-source-docs.net...

    focusj 評(píng)論0 收藏0
  • redux源碼解讀--applyMiddleware源碼解析

    摘要:的中間件主要是通過(guò)模塊實(shí)現(xiàn)的。返回的也是一個(gè)對(duì)象這個(gè)其實(shí)就是,各個(gè)中間件的最底層第三層的哪個(gè)函數(shù)組成的圓環(huán)函數(shù)構(gòu)成的這就是對(duì)源碼的一個(gè)整體解讀,水平有限,歡迎拍磚。后續(xù)的源碼解讀和測(cè)試?yán)涌梢躁P(guān)注源碼解讀倉(cāng)庫(kù) applyMiddleware源碼解析 中間件機(jī)制在redux中是強(qiáng)大且便捷的,利用redux的中間件我們能夠?qū)崿F(xiàn)日志記錄,異步調(diào)用等多種十分實(shí)用的功能。redux的中間件主要是...

    Atom 評(píng)論0 收藏0
  • 精讀《源碼學(xué)習(xí)》

    摘要:精讀原文介紹了學(xué)習(xí)源碼的兩個(gè)技巧,并利用實(shí)例說(shuō)明了源碼學(xué)習(xí)過(guò)程中可以學(xué)到許多周邊知識(shí),都讓我們受益匪淺。討論地址是精讀源碼學(xué)習(xí)如果你想?yún)⑴c討論,請(qǐng)點(diǎn)擊這里,每周都有新的主題,周末或周一發(fā)布。 1. 引言 javascript-knowledge-reading-source-code 這篇文章介紹了閱讀源碼的重要性,精讀系列也已有八期源碼系列文章,分別是: 精讀《Immer.js》源...

    aboutU 評(píng)論0 收藏0
  • 不一樣的redux源碼解讀

    摘要:這里還有一個(gè)疑問(wèn)點(diǎn)就是的嵌套,最開(kāi)始也我不明白,看了源碼才知道,這里返回的也是接受也就是一個(gè)所以可以正常嵌套。以作為參數(shù),調(diào)用上一步返回的函數(shù)以為參數(shù)進(jìn)行調(diào)用。 1、本文不涉及redux的使用方法,因此可能更適合使用過(guò) redux 的同學(xué)閱讀2、當(dāng)前redux版本為4.0.13、更多系列文章請(qǐng)看 Redux作為大型React應(yīng)用狀態(tài)管理最常用的工具。雖然在平時(shí)的工作中很多次的用到了它...

    Salamander 評(píng)論0 收藏0
  • redux源碼解讀--compose源碼解析

    摘要:源碼解析模塊的代碼十分簡(jiǎn)練,但是實(shí)現(xiàn)的作用卻是十分強(qiáng)大。只傳遞一個(gè)參數(shù)的時(shí)候,就直接把這個(gè)函數(shù)返回返回組合函數(shù)這就是對(duì)源碼的一個(gè)整體解讀,水平有限,歡迎拍磚。后續(xù)的源碼解讀和測(cè)試?yán)涌梢躁P(guān)注源碼解讀倉(cāng)庫(kù) compose源碼解析 compose模塊的代碼十分簡(jiǎn)練,但是實(shí)現(xiàn)的作用卻是十分強(qiáng)大。redux為何稱為redux?有人說(shuō)就是reduce和flux的結(jié)合體,而reduce正是comp...

    lk20150415 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<