欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談vuex為什么不建議在action中修改state

 更新時間:2020年02月02日 10:19:18   作者:快狗打車前端團(tuán)隊  
這篇文章主要介紹了淺談vuex為什么不建議在action中修改state,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

背景

在最近的一次需求開發(fā)過程中,有再次使用到Vuex,在狀態(tài)更新這一方面,我始終遵循著官方的“叮囑”,謹(jǐn)記“一定不要在action中修改state,而是要在mutation中修改”;于是我不禁產(chǎn)生了一個疑問:Vuex為什么要給出這個限制,它是基于什么原因呢?帶著這個疑問我查看Vuex的源碼,下面請大家跟著我的腳步,來一起揭開這個問題的面紗。

一起閱讀源碼吧~

1.首先我們可以在src/store.js這個文件的Store類中找到下面這段代碼

// ...
this.dispatch = function boundDispatch (type, payload) {
 return dispatch.call(store, type, payload)
}
this.commit = function boundCommit (type, payload, options) {
 return commit.call(store, type, payload, options)
}
// ...

上面是Vuex兩個最核心的API:dispatch & commit,它們是分別用來提交action和mutation的

那么既然我們今天的目的是為了“了解為什么不能在action中修改state”,所以我們就先看看mutation是怎樣修改state的,然而mutation是通過commit提交的,所以我們先看一下commit的內(nèi)部實(shí)現(xiàn)

commit&mutation

2.commit方法的核心代碼大致如下:

commit (_type, _payload, _options) {
  // ...
  this._withCommit(() => {
   entry.forEach(function commitIterator (handler) {
    handler(payload)
   })
  })
  // ...
}

不難看出,Vuex在commit(提交)某種類型的mutation時,會先用_withCommit包裹一下這些mutation,即作為參數(shù)傳入_withCommit;那么我們來看看_withCommit的內(nèi)部實(shí)現(xiàn)(ps:這里之所以說”某種類型的mutation“,是因?yàn)閂uex的確支持聲明多個同名的mutation,不過前提是它們在不同的namespace下;action同理)

3._withCommit方法的代碼如下:

_withCommit (fn) {
  const committing = this._committing
  this._committing = true
  fn()
  this._committing = committing
 }

是的,你沒有看錯,它真的只有4行代碼;這里我們注意到有一個標(biāo)志位_committing,在執(zhí)行fn前,這個標(biāo)志位會被置為true,這個點(diǎn)我們先記下,一會兒會用到

4.接下來,我要為大家要介紹的是resetStoreVM這個函數(shù),它的作用是初始化store,它首次被執(zhí)行是在Store的構(gòu)造函數(shù)中

function resetStoreVM (store, state, hot) {
 // ...
 if (store.strict) {
  enableStrictMode(store)
 }
// ...
}

在這里有一處需要我們注意:resetStoreVM對strict(是否啟用嚴(yán)格模式)做了判斷,這里假設(shè)我們啟用嚴(yán)格模式,那么就會執(zhí)行enableStrictMode這個函數(shù),下面繼續(xù)來看看它的內(nèi)部實(shí)現(xiàn)

function enableStrictMode (store) {
 store._vm.$watch(function () { return this._data.$$state }, () => {
  if (process.env.NODE_ENV !== 'production') {
   assert(store._committing, `do not mutate vuex store state outside mutation handlers.`)
  }
 }, { deep: true, sync: true })
}

這里對Vue組件實(shí)例的state做了監(jiān)聽,一旦監(jiān)聽到變化,就會執(zhí)行asset(斷言),它斷言的恰巧就是剛才我讓大家記住的那個_committing標(biāo)志位,那么我們再來看看這個asset做了些什么

5.asset方法在src/util.js這個文件中

export function assert (condition, msg) {
 if (!condition) throw new Error(`[vuex] ${msg}`)
}

這個方法很簡單,就是判斷第一個參數(shù)是否為truly值,如果不為真,就拋出一個異常
到此,我們已簡單地了解了commit和mutation的邏輯,下面再來看看dispatch和action

dispatch&action

6.dispatch代碼大致如下:

dispatch (_type, _payload) {
  const {
   type,
   payload
  } = unifyObjectStyle(_type, _payload)

  const action = { type, payload }
  const entry = this._actions[type]

 // ...
  const result = entry.length > 1
   ? Promise.all(entry.map(handler => handler(payload)))
   : entry[0](payload)
 // ...
 }

這里我們注意到,當(dāng)某種類型的action只有一個聲明時,action的回調(diào)會被當(dāng)作普通函數(shù)執(zhí)行,而當(dāng)如果有多個聲明時,它們是被視為Promise實(shí)例,并且用Promise.all執(zhí)行,總所周知,Promise.all在執(zhí)行Promise時是不保證順序的,也就是說,假如有3個Promise實(shí)例:P1、P2、P3,它們3個之中不一定哪個先有返回結(jié)果,那么我們仔細(xì)思考一下:如果同時在多個action中修改了同一個state,那會有什么樣的結(jié)果?

其實(shí)很簡單,我們在多個action中修改同一個state,因?yàn)楹苡锌赡苊總€action賦給state的新值都有所不同,并且不能保證最后一個有返回結(jié)果action是哪一個action,所以最后賦予state的值可能是錯誤的

那么Vuex為什么要使用Promise.all執(zhí)行action呢?其實(shí)也是出于性能考慮,這樣我們就可以最大限度進(jìn)行異步操作并發(fā)
眼尖的同學(xué)可能已經(jīng)發(fā)現(xiàn)在dispatch中并沒有看到_committing的身影,就是Vuex對action修改state的限制:當(dāng)action想要修改state時,因?yàn)開committing沒有事先被置為true,而導(dǎo)致asset階段無法通過

但這個限制只限于開發(fā)階段,因?yàn)樵趀nableStrictMode函數(shù)中,Webpack加入了對環(huán)境的判斷,如果不是生產(chǎn)環(huán)境(也就是開發(fā)環(huán)境)才會輸出asset(斷言)這行代碼

function enableStrictMode (store) {
 store._vm.$watch(function () { return this._data.$$state }, () => {
  if (process.env.NODE_ENV !== 'production') {
   assert(store._committing, `do not mutate vuex store state outside mutation handlers.`)
  }
 }, { deep: true, sync: true })
}

那么也就是說如果你強(qiáng)行在生產(chǎn)環(huán)境中用action修改state,Vuex也不會阻止你,它可能僅僅是給你一個警告;而且按道理來說,如果我們能夠保證同一類型的action只有一個聲明,那么無論是使用action還是mutation來修改state結(jié)果都是一樣的,因?yàn)閂uex針對這種情況,沒有使用Promise.all執(zhí)行action,所以也就不會存在返回結(jié)果先后問題

dispatch (_type, _payload) {
  // ...
  const result = entry.length > 1
   ? Promise.all(entry.map(handler => handler(payload)))
   : entry[0](payload)
  // ...
 }

但是凡是靠人遵守的約定都是不靠譜的,所以我們在平時使用Vuex時,最好還是遵守官方的約束,否則線上代碼有可能出現(xiàn)bug,這不是我們所期望的。

結(jié)束語

Vuex這一限制其實(shí)也是出于代碼設(shè)計考慮,action和mutation各司其事,本質(zhì)上也是遵守了“單一職責(zé)”原則。以上就是我對“Vuex為什么不允許在action中修改狀態(tài)“這個問題的分析,希望對大家有所幫助,也歡迎指正

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Vue中computed和watch的區(qū)別小結(jié)

    Vue中computed和watch的區(qū)別小結(jié)

    watch和computed都是以Vue的依賴追蹤機(jī)制為基礎(chǔ)的,當(dāng)某一個依賴型數(shù)據(jù)發(fā)生變化的時候,所有依賴這個數(shù)據(jù)的相關(guān)數(shù)據(jù)會自動發(fā)生變化,即自動調(diào)用相關(guān)的函數(shù),來實(shí)現(xiàn)數(shù)據(jù)的變動,這篇文章簡單介紹下Vue中computed和watch的區(qū)別異同,感興趣的朋友一起看看吧
    2022-12-12
  • vue中自定義右鍵菜單插件

    vue中自定義右鍵菜單插件

    這篇文章主要為大家詳細(xì)介紹了vue中自定義右鍵菜單插件,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-04-04
  • vue使用技巧及vue項目中遇到的問題

    vue使用技巧及vue項目中遇到的問題

    這篇文章主要介紹了vue使用技巧及vue項目中遇到的問題,本文給大家?guī)淼闹皇且徊糠?,后續(xù)還會持續(xù)更新,感興趣的朋友跟隨腳本之家小編一起學(xué)習(xí)吧
    2018-06-06
  • el-date-picker日期范圍限制的實(shí)現(xiàn)

    el-date-picker日期范圍限制的實(shí)現(xiàn)

    本文主要介紹了el-date-picker日期范圍限制的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-05-05
  • vue中el表單的簡單查詢方法

    vue中el表單的簡單查詢方法

    本文主要介紹了vue中el表單的簡單查詢方法,主要包括表單頁面根據(jù)groupid 、type 、errortype進(jìn)行數(shù)據(jù)過濾,感興趣的可以了解一下
    2023-10-10
  • vue生成token保存在客戶端localStorage中的方法

    vue生成token保存在客戶端localStorage中的方法

    本篇文章主要介紹了vue生成token保存在客戶端localStorage中的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-10-10
  • Vue-resource實(shí)現(xiàn)ajax請求和跨域請求示例

    Vue-resource實(shí)現(xiàn)ajax請求和跨域請求示例

    本篇文章主要介紹了Vue-resource實(shí)現(xiàn)ajax請求和跨域請求示例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-02-02
  • 如何使用sm4js進(jìn)行加密和國密sm4總結(jié)

    如何使用sm4js進(jìn)行加密和國密sm4總結(jié)

    近期由于公司項目的需要開始研究國密SM4加密,下面這篇文章主要給大家介紹了關(guān)于如何使用sm4js進(jìn)行加密和國密sm4的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-04-04
  • Vue 2.0在IE11中打開項目頁面空白的問題解決

    Vue 2.0在IE11中打開項目頁面空白的問題解決

    這篇文章主要給大家介紹了關(guān)于Vue 2.0在ie 11中打開項目頁面空白問題的解決方法,文中詳細(xì)分析出現(xiàn)該問題的原因,并給出了詳細(xì)的解決方法,需要的朋友可以參考借鑒,下面跟著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-07-07
  • 在Vue中如何使用Cookie操作實(shí)例

    在Vue中如何使用Cookie操作實(shí)例

    這篇文章主要介紹了在Vue中如何使用Cookie操作實(shí)例的相關(guān)資料,需要的朋友可以參考下
    2017-07-07

最新評論