亚洲中字慕日产2020,大陆极品少妇内射AAAAAA,无码av大香线蕉伊人久久,久久精品国产亚洲av麻豆网站

資訊專欄INFORMATION COLUMN

重讀redux源碼(二)

dingda / 3250人閱讀

摘要:函數(shù)組合,科里化的串聯(lián)結(jié)合示例源碼,實現(xiàn)也很優(yōu)雅,對于返回的,將等參數(shù)傳遞進去,然后執(zhí)行,等待回調(diào)異步完成再。對于正常對象則進行下一步。

前言

作為前端狀態(tài)管理器,這個比較跨時代的工具庫redux有很多實現(xiàn)和思想值得我們思考。在深入源碼之前,我們可以相關(guān)注下一些常見問題,這樣帶著問題去看實現(xiàn),也能更加清晰的了解。

常見問題

大概看了下主要有這么幾個:

    redux三大原則
    這個可以直接參考官方文檔

    redux 的優(yōu)缺點。 關(guān)于優(yōu)缺點,太主觀了大家見仁見智。

    redux中間件相關(guān),洋蔥模型是什么,常見中間件。

背景

有關(guān)acton,reducer相關(guān)的部分可以看我前面的文章。我們主要關(guān)注針對store和中間件相關(guān)的部分來解讀。

store的創(chuàng)建

作為維護和管理數(shù)據(jù)的部分,store在redux中的作用十分重要。在action發(fā)出指令,reduxer進行數(shù)據(jù)更新之后,監(jiān)聽數(shù)據(jù)變化和同步數(shù)據(jù)更新的操作都要借助store來實現(xiàn)。

createStore 輸入和輸出

首先看下createStore的使用,即常見的就是接受經(jīng)過combineReducers處理之后的reducer和初始的state

import reducer from "./reducers"
const store = createStore(reducer,initialState)

此外還可以接受第三個參數(shù)enhancer(增強器,一般就是applyMiddleware)

/**
 * 創(chuàng)建管理state 樹的Redux store
 * 應(yīng)用中只能存在一個store,為了區(qū)分不同action對應(yīng)的reducer,
 * 可以通過combineReducers來關(guān)聯(lián)不同的reducer
 * @param {Function} reducer   combineReducers關(guān)聯(lián)之后的reducer
 * @param {Object} preloadedState 初始state
 * @param {Function} enhancer 可以增強store功能的函數(shù),例如中間件等。唯一適合
 * @returns 返回一個Store 以維護state和監(jiān)聽變化  
 */
export default function createStore(reducer, preloadedState, enhancer) {
  // 如果第二個參數(shù)為func,redux認為忽略了初始state,而是
  if (typeof preloadedState === "function" && typeof enhancer === "undefined") {
    // enhancer增強劑,即利用中間件等來增強redux能力
    enhancer = preloadedState
    preloadedState = undefined
  }
  // 返回具有dispatch等屬性的對象 即store
  return {
    dispatch,
    subscribe,
    getState,
    replaceReducer,
    [$$observable]: observable
  }
 } 

按照一般的執(zhí)行順序,我們先看下對于參數(shù)的處理(平時大家也是一樣,一個函數(shù),執(zhí)行之前盡量判斷入?yún)⑹欠穹项A(yù)期,避免直接處理造成的錯誤)

入?yún)⑻幚?/b>

對于三個參數(shù),后兩個是非必填的,但如果第二個參數(shù)是function,reduxe認為其實encher,不然初始狀態(tài)是個函數(shù)不符合redux的預(yù)期,只能這樣處理了。

// 如果第二個參數(shù)為func,redux認為忽略了初始state,而是
  if (typeof preloadedState === "function" && typeof enhancer === "undefined") {
    // enhancer增強劑,即利用中間件等來增強redux能力
    enhancer = preloadedState
    preloadedState = undefined
  }

  if (typeof enhancer !== "undefined") {
    if (typeof enhancer !== "function") {
      throw new Error("Expected the enhancer to be a function.")
    }
    // 對于存在的enhancer,高階函數(shù)函數(shù)的用法,
    // 接受createStore返回一個增加功能之后的函數(shù),然后再傳入相關(guān)reducer得到store。
    return enhancer(createStore)(reducer, preloadedState)
  }

  if (typeof reducer !== "function") {
    throw new Error("Expected the reducer to be a function.")
  }
  // 一切符合預(yù)期,沒有 enhancer,那么初始賦值
  let currentReducer = reducer
  let currentState = preloadedState
  let currentListeners = []
  // 監(jiān)聽隊列
  let nextListeners = currentListeners
  // dispatch標(biāo)識
  let isDispatching = false
  
  // 初始狀態(tài)更新之后,聲明init狀態(tài)完成。
  dispatch({ type: ActionTypes.INIT })

dispatch的實現(xiàn)

dispatch的作用就是根據(jù)action,執(zhí)行對應(yīng)reducer以更新state。并執(zhí)行監(jiān)聽隊列。
下面就來看dispatch的用法和實現(xiàn)。 常見使用:

// redux要求 參數(shù)必須為純對象
dispatch({ type: ActionTypes.INIT })

那么對于純對象,redux做了些什么呢

 /**
   * 通知方法,參數(shù)為純的js對象,標(biāo)明更新內(nèi)容
   * @param {Object} action 
   */
  function dispatch(action) {
    // 是否滿足純凈對象
    if (!isPlainObject(action)) {
      throw new Error(
        "省略"
      )
    }
    // 必須的type是否存在 
    if (typeof action.type === "undefined") {
      throw new Error(
        "省略"
      )
    }
    // 判斷是否處于某個action的dispatch中,大家一起dispatch可能死循環(huán)
    if (isDispatching) {
      throw new Error("Reducers may not dispatch actions.")
    }

    try {
      // 開始dispatch,加鎖,標(biāo)明狀態(tài)
      isDispatching = true
      // 將當(dāng)前狀態(tài)和更新action,傳給當(dāng)前reducer處理
      // 這里可以回想下我們reducer中的兩個參數(shù),state和action 對應(yīng)的是什么
      /**
       * const todos = (state = [], action) => {
       */
      currentState = currentReducer(currentState, action)
    } finally {
      // 有異常,鎖置為false,不影響別的dispatch
      isDispatching = false
    }
    // 執(zhí)行dispatch,并且更新當(dāng)前監(jiān)聽隊列為 最新隊列
    const listeners = (currentListeners = nextListeners)
    // 依次執(zhí)行,監(jiān)聽器
    for (let i = 0; i < listeners.length; i++) {
      const listener = listeners[i]
      listener()
    }

    return action
  }

createStore初始化完成之后會執(zhí)行dispatch({ type: ActionTypes.INIT }),此時執(zhí)行初始化操作。

我們要關(guān)注下currentState的計算,
將currentState,action傳給reducer處理,然后更新currentState。

針對初始化來說currentState其實就是initState:

// 初始化狀態(tài)
let currentState = preloadedState
/****省略***/
// 這里可以回想下我們reducer中的兩個參數(shù),state和action對應(yīng)的值
currentState = currentReducer(currentState, action)

reducer示例:

const todos = (state = [], action) => {
switch (action.type) {
    case "ADD_TODO":
      return [
        ...state,
        {
          id: action.id,
          text: action.text,
          completed: false
        }
      ]
    }
getSate實現(xiàn)

getState就是獲得store的state。這個比較簡單。當(dāng)結(jié)合react-redux使用時,其會幫我們進行操作。我們就不用自行調(diào)用這個方法了,所以不要疑惑從哪里獲取的state。

/**
   * 返回應(yīng)用當(dāng)前狀態(tài)
   * 不過需要看下當(dāng)前是否有更新正在進行,是的話則提示
   */
  function getState() {
    // 判斷是否isDispatching 中
    if (isDispatching) {
      throw new Error("省略")
    }
    return currentState
  }
subscribe

subscribe是比較重要的一個方法,用來供我們監(jiān)聽狀態(tài)的變化,以執(zhí)行相關(guān)操作。
例如react-redux中的handleChange 會對是否pure組件及state進行對比,以提升渲染效率。

示例:

 this.unsubscribe = this.store.subscribe(this.handleChange.bind(this))

實現(xiàn): 返回的是一個函數(shù),可以進行unsubscribe操作。

/** 
   * 訂閱通知
   */
  function subscribe(listener) {
    if (typeof listener !== "function") {
      throw new Error("Expected the listener to be a function.")
    }

    if (isDispatching) {
      throw new Error(
        "省略"
      )
    }
    // 是否已經(jīng)監(jiān)聽過
    let isSubscribed = true
    // 監(jiān)聽隊列是否相同,區(qū)分開,操作nextListeners
    ensureCanMutateNextListeners()
    // 新增監(jiān)聽事件
    nextListeners.push(listener)

    return function unsubscribe() {
      if (!isSubscribed) {
        return
      }

      if (isDispatching) {
        throw new Error(
          "省略"
        )
      }
      // 注冊完成,可以進行取消操作
      isSubscribed = false
      // 保持事件隊列 同步
      ensureCanMutateNextListeners()
      const index = nextListeners.indexOf(listener)
      // 刪除監(jiān)聽事件
      nextListeners.splice(index, 1)
    }
  }

replaceReducer

這個開發(fā)比較少用,用于熱更新

// 用于reducer的熱替換,開發(fā)中一般不會直接使用
  function replaceReducer(nextReducer) {
    if (typeof nextReducer !== "function") {
      throw new Error("Expected the nextReducer to be a function.")
    }

    currentReducer = nextReducer
    // 更新值之后,進行dispatch。
    dispatch({ type: ActionTypes.REPLACE })
  }

到這里createStore已經(jīng)解析完成了,大家應(yīng)該了解該方法到底做了些什么操作吧。
簡單概括一下就是:接收reducer和initState,返回一個store 對象。該對象提供了監(jiān)聽、分發(fā)等功能,以實現(xiàn)數(shù)據(jù)的更新。

實際使用中的問題

經(jīng)過上面的解讀之后,對于redux的常規(guī)應(yīng)用應(yīng)該有所了解了。不過實際使用中可能會遇到些問題。
例如action要求是純對象,而我們獲取數(shù)據(jù)一般是異步的,這就需要借助redux-thunk這個中間件了。
actionCreater返回一個函數(shù)。如下:

export function func1() {
  return dispatch => {
      dispatch({
      type:"test",
      data:"a"
      })
  }
}

在了解如何實現(xiàn)之前,需要先看下redux中間件的原理。 因為reducer更多的關(guān)注的是數(shù)據(jù)的操作,對于一些公共的方法,需要抽離出來,不過這些方法在何時使用呢,redux為我們提供了中間件來滿足需求。

redux中間件原理

redux 借鑒了 Koa里 middleware 的思想,即鼎鼎大名的洋蔥模型。

不過這里請求對應(yīng)的是dispatch的過程。

每次dispatch的過程中,都要依次將中間件執(zhí)行一遍。
遇到阻塞或者操作完成,執(zhí)行下個中間件,直到執(zhí)行完成,以便我們事先日志,監(jiān)控、異常上報等功能。
那么redux 又是如何支持中間件的呢。這就離不開applyMiddleware了。
這里前面的

applyMiddleware實現(xiàn)思路

實現(xiàn)思想比較簡單,通過科里化和compose,為符合規(guī)范的中間件分配訪問dispatch和store的途徑,以便在不同階段來自定義數(shù)據(jù)更新。
例如異步操作,返回的不是對象,那么就執(zhí)行返回的函數(shù),然后調(diào)用下一個中間件。等異步請求結(jié)束,再次dispatch 對應(yīng)的action。

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.`
      )
    }
    // 賦予每個中間件訪問store的能力。
    const middlewareAPI = {
      getState: store.getState,
      // 箭頭函數(shù)保存dispatch,保證其的同步更新
      dispatch: (...args) => dispatch(...args)
    }
    // 串聯(lián)中間件,并賦予每個中間件訪問dispatch的能力。
    const chain = middlewares.map(middleware => middleware(middlewareAPI))
    // 關(guān)聯(lián)dispatch與中間件,組合調(diào)用之后得到類似下面的新對象
    // dispatch = f1(f2(f3(store.dispatch))));
    dispatch = compose(...chain)(store.dispatch)
    

    return {
      ...store,
      dispatch
    }
  }
}

這樣執(zhí)行之后返回的,對象就是增強之后的store了。

compose的實現(xiàn)

redux中compose是柯里化函數(shù)的一個示例,目的是將函數(shù)串聯(lián)起來。

/**
 * 函數(shù)組合,科里化的串聯(lián)
 */
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)))
}
結(jié)合redux-thunk示例

redux-thunk源碼,實現(xiàn)也很優(yōu)雅,對于返回的function,將dispatch等參數(shù)傳遞進去,然后執(zhí)行,等待回調(diào)異步完成再dispatch。對于正常對象則進行下一步。

function createThunkMiddleware(extraArgument) {
  return ({ dispatch, getState }) => next => action => {
    // 每次dispatch的時候都會進行判斷,如果是個函數(shù),那就執(zhí)行函數(shù),不再進行下一步吧,這樣就避免了,函數(shù)不滿足action要求的問題
    if (typeof action === "function") {
      return action(dispatch, getState, extraArgument);
    }

    return next(action);
  };
}

const thunk = createThunkMiddleware();
thunk.withExtraArgument = createThunkMiddleware;

export default thunk;

那么實際使用時,在createStore時加入該中間件即可:

import { createStore, applyMiddleware } from "redux"
import thunk from "redux-thunk";
const store = createStore(
  reducer,
  applyMiddleware({
  ...middleware,
  thunk})
)

那么到這里對于redux的中間件 也就是問題2,我想大家也比較清楚了。
對于常見中間件可以參考

結(jié)束語 參考文章

redux中文文檔
深入React技術(shù)棧
加上重讀redux源碼一和帶著問題看 react-redux 源碼實現(xiàn)總算將redux及react-redux重讀了一遍。可能有人會說道這些源碼,看完也會忘,有這個必要嗎。我感覺分情況來看,如果我們只是使用,那么看官方文檔就可以了,當(dāng)遇到某些疑問好像找不到貼切解釋的時候,不放一看。
此外也是學(xué)習(xí)大佬們的設(shè)計思路和實現(xiàn)方式,有的放矢才能開卷有益。

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

轉(zhuǎn)載請注明本文地址:http://www.ezyhdfw.cn/yun/6973.html

相關(guān)文章

  • React生態(tài),dva源碼閱讀

    摘要:下面會從淺到深,淡淡在閱讀源碼過程中自己的理解。分拆子頁面后,每一個子頁面對應(yīng)一個文件??偨Y(jié)上面就是最早版本的源碼,很簡潔的使用了等其目的也很簡單簡化相關(guān)生態(tài)的繁瑣邏輯參考源碼地址 ??dva的思想還是很不錯的,大大提升了開發(fā)效率,dva集成了Redux以及Redux的中間件Redux-saga,以及React-router等等。得益于Redux的狀態(tài)管理,以及Redux-saga中...

    bergwhite 評論0 收藏0
  • React生態(tài),dva源碼閱讀

    摘要:下面會從淺到深,淡淡在閱讀源碼過程中自己的理解。分拆子頁面后,每一個子頁面對應(yīng)一個文件??偨Y(jié)上面就是最早版本的源碼,很簡潔的使用了等其目的也很簡單簡化相關(guān)生態(tài)的繁瑣邏輯參考源碼地址 ??dva的思想還是很不錯的,大大提升了開發(fā)效率,dva集成了Redux以及Redux的中間件Redux-saga,以及React-router等等。得益于Redux的狀態(tài)管理,以及Redux-saga中...

    txgcwm 評論0 收藏0
  • React生態(tài),dva源碼閱讀

    摘要:下面會從淺到深,淡淡在閱讀源碼過程中自己的理解。分拆子頁面后,每一個子頁面對應(yīng)一個文件??偨Y(jié)上面就是最早版本的源碼,很簡潔的使用了等其目的也很簡單簡化相關(guān)生態(tài)的繁瑣邏輯參考源碼地址 ??dva的思想還是很不錯的,大大提升了開發(fā)效率,dva集成了Redux以及Redux的中間件Redux-saga,以及React-router等等。得益于Redux的狀態(tài)管理,以及Redux-saga中...

    harryhappy 評論0 收藏0
  • 重讀《學(xué)習(xí)JavaScript數(shù)據(jù)結(jié)構(gòu)與算法-第三版》- 第3章 數(shù)組(

    摘要:定場詩守法朝朝憂悶,強梁夜夜歡歌損人利己騎馬騾,正值公平挨餓修橋補路瞎眼,殺人放火兒多我到西天問我佛,佛說我也沒轍前言讀學(xué)習(xí)數(shù)據(jù)結(jié)構(gòu)與算法第章數(shù)組,本小節(jié)將繼續(xù)為各位小伙伴分享數(shù)組的相關(guān)知識數(shù)組的新功能。 定場詩 守法朝朝憂悶,強梁夜夜歡歌; 損人利己騎馬騾,正值公平挨餓; 修橋補路瞎眼,殺人放火兒多; 我到西天問我佛,佛說:我也沒轍! 前言 讀《學(xué)習(xí)JavaScript數(shù)據(jù)結(jié)構(gòu)與算法...

    William_Sang 評論0 收藏0
  • 重讀 Gulp

    摘要:當(dāng)接收一個回調(diào)函數(shù)的時候,一定要注意回調(diào)函數(shù)中的參數(shù)。主要作用就是用來讀取文件或者文件夾中的數(shù)據(jù)。表示文件的名稱指的是發(fā)生的變化使用技巧的進一步使用,可以參照中文官網(wǎng)中的技巧集。 Gulp 簡介 Gulp 對現(xiàn)在的前端而言,是一個稍微老舊的工具了,但是,為了復(fù)習(xí)以前學(xué)過的內(nèi)容,還是把它翻出來,放在自己的博客中。說不定哪天又用到了呢。 需要說明的是,這里使用的 Gulp 版本是 3.9....

    vpants 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<