vue源碼(3):vue異步更新

問題:

1.能簡單說一下vue 的異步更新機制嗎乏梁?

2.nextTick的原理是什么次洼?

  • dep.notify

  • 源碼地址:/src/core/observer/dep.js
/*
   *通知dep中所有的watcher,執(zhí)行watcher.updata()方法
  */ 
  notify () {
    const subs = this.subs.slice()
    if (process.env.NODE_ENV !== 'production' && !config.async) {
      subs.sort((a, b) => a.id - b.id)
    }
    //遍歷dep中的存儲就的watcher遇骑,執(zhí)行watcher.updata()
    for (let i = 0, l = subs.length; i < l; i++) {
      subs[i].update()
    }
  • watcher.update

  • 源碼地址:/src/core/observer/watcher.js
  /**
   * 根據(jù)watcher配置項卖毁,決定接下來怎么走,一般是queryWatcher
   * 
   */
  update () {
    /* istanbul ignore else */
    if (this.lazy) {
      // 懶執(zhí)行時走這里落萎,比如computed
      // 將dirty職位true亥啦,可以讓computedGetter執(zhí)行重新計算computed回調(diào)函數(shù)的執(zhí)行結(jié)果
      this.dirty = true
    } else if (this.sync) {
      // 同步執(zhí)行,使用vm.watch選項是可以傳遞一個sync選項练链,
      // 當(dāng)為true時翔脱,并且數(shù)據(jù)更新時watcher就不走異步隊列,直接執(zhí)行this.run
      this.run()
    } else {
      // 更新時一般都走這里媒鼓,將watcher放入隊列watcher中
      queueWatcher(this)
    }
  }
  • queueWatcher

  • 源碼地址:/src/core/observer/scheduler.js
/**
 * 將watcher放入届吁,watcher隊列中
 */
export function queueWatcher (watcher: Watcher) {
  //給每個watcher添加id
  const id = watcher.id
  //判重watcher不會重復(fù)入隊
  if (has[id] == null) {
    // 緩存一下watcher,用于判斷watcher是否已入隊
    has[id] = true
    if (!flushing) {
      // 如果flushing=false绿鸣,表示當(dāng)前watcher沒有被刷新疚沐,watcher可以直接入隊
      queue.push(watcher)
    } else {
      // watcher隊列已經(jīng)被刷新,這個時候watcher入隊就需要特殊操作
      // 保證watcher入隊以后潮模,刷新的watcher隊列為有序的
      let i = queue.length - 1
      while (i > index && queue[i].id > watcher.id) {
        i--
      }
      queue.splice(i + 1, 0, watcher)
    }
    // queue the flush
    if (!waiting) {
      // waiting為false時亮蛔,表示當(dāng)前瀏覽器的異步隊列任務(wù)不支持flushSchedulerQueue函數(shù)
      waiting = true

      if (process.env.NODE_ENV !== 'production' && !config.async) {
        // 同步執(zhí)行,直接刷新watcher隊列
        // 性能會大打折扣
        flushSchedulerQueue()
        return
      }
      nextTick(flushSchedulerQueue)
    }
  }
}
  • nextTick

  • 源碼地址:/src/core/util/next-tick.js
/**
 * 完成兩件事:
 *   1擎厢、用 try catch 包裝 flushSchedulerQueue 函數(shù)究流,然后將其放入 callbacks 數(shù)組
 *   2辣吃、如果 pending 為 false,表示現(xiàn)在瀏覽器的任務(wù)隊列中沒有 flushCallbacks 函數(shù)
 *     如果 pending 為 true芬探,則表示瀏覽器的任務(wù)隊列中已經(jīng)被放入了 flushCallbacks 函數(shù)齿尽,
 *     待執(zhí)行 flushCallbacks 函數(shù)時,pending 會被再次置為 false灯节,表示下一個 flushCallbacks 函數(shù)可以進入
 *     瀏覽器的任務(wù)隊列了
 * pending 的作用:保證在同一時刻循头,瀏覽器的任務(wù)隊列中只有一個 flushCallbacks 函數(shù)
 * @param {*} cb 接收一個回調(diào)函數(shù) => flushSchedulerQueue
 * @param {*} ctx 上下文
 * @returns 
 */
export function nextTick (cb?: Function, ctx?: Object) {
  let _resolve
  // 用 callbacks 數(shù)組存儲經(jīng)過包裝的 cb 函數(shù)
  callbacks.push(() => {
    if (cb) {
      // 用 try catch 包裝回調(diào)函數(shù),便于錯誤捕獲
      try {
        cb.call(ctx)
      } catch (e) {
        handleError(e, ctx, 'nextTick')
      }
    } else if (_resolve) {
      _resolve(ctx)
    }
  })
  if (!pending) {
    pending = true
    // 執(zhí)行 timerFunc炎疆,在瀏覽器的任務(wù)隊列中(首選微任務(wù)隊列)放入 flushCallbacks 函數(shù)
    timerFunc()
  }
  // $flow-disable-line
  if (!cb && typeof Promise !== 'undefined') {
    return new Promise(resolve => {
      _resolve = resolve
    })
  }
}
  • flushCallbacks

  • 源碼地址:/src/core/util/next-tick.js
/**
* 做了三件事: 
*      1.將pending設(shè)置為false
*      2.清空callbacks數(shù)組
*      3.執(zhí)行 callbacks 數(shù)組中的每一個函數(shù)(比如 flushSchedulerQueue卡骂、用戶調(diào)用 nextTick 傳遞的回調(diào)函數(shù))
*/
const callbacks = []
let pending = false

function flushCallbacks () {
  pending = false
  const copies = callbacks.slice(0)
  callbacks.length = 0
  // 遍歷 callbacks 數(shù)組,執(zhí)行其中存儲的每個 flushSchedulerQueue 函數(shù)/
  for (let i = 0; i < copies.length; i++) {
    copies[i]()
  }
}
  • flushSchedulerQueue

  • 源碼地址:/src/core/observer/scheduler.js
/**
 * Flush both queues and run the watchers.
 * 刷新隊列形入,由 flushCallbacks 函數(shù)負責(zé)調(diào)用全跨,主要做了如下兩件事:
 *   1、更新 flushing 為 ture亿遂,表示正在刷新隊列浓若,在此期間往隊列中 push 新的 watcher 時需要特殊處理(將其放在隊列的合適位置)
 *   2、按照隊列中的 watcher.id 從小到大排序蛇数,保證先創(chuàng)建的 watcher 先執(zhí)行挪钓,也配合 第一步
 *   3、遍歷 watcher 隊列耳舅,依次執(zhí)行 watcher.before碌上、watcher.run,并清除緩存的 watcher
 */
function flushSchedulerQueue () {
  currentFlushTimestamp = getNow()
  // 標(biāo)志現(xiàn)在正在刷新隊列
  flushing = true
  let watcher, id

  /**
   * 刷新隊列之前先給隊列排序(升序)浦徊,可以保證:
   *   1馏予、組件的更新順序為從父級到子級,因為父組件總是在子組件之前被創(chuàng)建
   *   2盔性、一個組件的用戶 watcher 在其渲染 watcher 之前被執(zhí)行霞丧,因為用戶 watcher 先于 渲染 watcher 創(chuàng)建
   *   3、如果一個組件在其父組件的 watcher 執(zhí)行期間被銷毀冕香,則它的 watcher 可以被跳過
   * 排序以后在刷新隊列期間新進來的 watcher 也會按順序放入隊列的合適位置
   */
  queue.sort((a, b) => a.id - b.id)

  // 這里直接使用了 queue.length蛹尝,動態(tài)計算隊列的長度,沒有緩存長度暂筝,是因為在執(zhí)行現(xiàn)有 watcher 期間隊列中可能會被 push 進新的 watcher
  for (index = 0; index < queue.length; index++) {
    watcher = queue[index]
    // 執(zhí)行 before 鉤子箩言,在使用 vm.$watch 或者 watch 選項時可以通過配置項(options.before)傳遞
    if (watcher.before) {
      watcher.before()
    }
    // 將緩存的 watcher 清除
    id = watcher.id
    has[id] = null

    // 執(zhí)行 watcher.run,最終觸發(fā)更新函數(shù)焕襟,比如 updateComponent 或者 獲取 this.xx(xx 為用戶 watch 的第二個參數(shù)),當(dāng)然第二個參數(shù)也有可能是一個函數(shù)饭豹,那就直接執(zhí)行
    watcher.run()
  }

  // keep copies of post queues before resetting state
  const activatedQueue = activatedChildren.slice()
  const updatedQueue = queue.slice()

  /**
   * 重置調(diào)度狀態(tài):
   *   1鸵赖、重置 has 緩存對象务漩,has = {}
   *   2、waiting = flushing = false它褪,表示刷新隊列結(jié)束
   *     waiting = flushing = false饵骨,表示可以像 callbacks 數(shù)組中放入新的 flushSchedulerQueue 函數(shù),并且可以向瀏覽器的任務(wù)隊列放入下一個 flushCallbacks 函數(shù)了
   */
  resetSchedulerState()

  // call component updated and activated hooks
  callActivatedHooks(activatedQueue)
  callUpdatedHooks(updatedQueue)

  // devtool hook
  /* istanbul ignore if */
  if (devtools && config.devtools) {
    devtools.emit('flush')
  }
}

/**
 * Reset the scheduler's state.
 */
function resetSchedulerState () {
  index = queue.length = activatedChildren.length = 0
  has = {}
  if (process.env.NODE_ENV !== 'production') {
    circular = {}
  }
  waiting = flushing = false
}

  • watcher.run

  • 源碼地址:/src/core/observer/watcher.js
/**
 * 由 刷新隊列函數(shù) flushSchedulerQueue 調(diào)用茫打,如果是同步 watch居触,則由 this.update 直接調(diào)用,完成如下幾件事:
 *   1老赤、執(zhí)行實例化 watcher 傳遞的第二個參數(shù)轮洋,updateComponent 或者 獲取 this.xx 的一個函數(shù)(parsePath 返回的函數(shù))
 *   2、更新舊值為新值
 *   3抬旺、執(zhí)行實例化 watcher 時傳遞的第三個參數(shù)弊予,比如用戶 watcher 的回調(diào)函數(shù)
 */
run () {
  if (this.active) {
    // 調(diào)用 this.get 方法
    const value = this.get()
    if (
      value !== this.value ||
      // Deep watchers and watchers on Object/Arrays should fire even
      // when the value is the same, because the value may
      // have mutated.
      isObject(value) ||
      this.deep
    ) {
      // 更新舊值為新值
      const oldValue = this.value
      this.value = value

      if (this.user) {
        // 如果是用戶 watcher,則執(zhí)行用戶傳遞的第三個參數(shù) —— 回調(diào)函數(shù)开财,參數(shù)為 val 和 oldVal
        try {
          this.cb.call(this.vm, value, oldValue)
        } catch (e) {
          handleError(e, this.vm, `callback for watcher "${this.expression}"`)
        }
      } else {
        // 渲染 watcher汉柒,this.cb = noop,一個空函數(shù)
        this.cb.call(this.vm, value, oldValue)
      }
    }
  }
}

  • watcher.get

  • 源碼地址:/src/core/observer/watcher.js
  /**
   * 執(zhí)行 this.getter责鳍,并重新收集依賴
   * this.getter 是實例化 watcher 時傳遞的第二個參數(shù)碾褂,一個函數(shù)或者字符串,比如:updateComponent 或者 parsePath 返回的函數(shù)
   * 為什么要重新收集依賴历葛?
   *   因為觸發(fā)更新說明有響應(yīng)式數(shù)據(jù)被更新了斋扰,但是被更新的數(shù)據(jù)雖然已經(jīng)經(jīng)過 observe 觀察了,但是卻沒有進行依賴收集啃洋,
   *   所以传货,在更新頁面時,會重新執(zhí)行一次 render 函數(shù)宏娄,執(zhí)行期間會觸發(fā)讀取操作问裕,這時候進行依賴收集
   */
  get () {
    // 打開 Dep.target,Dep.target = this
    pushTarget(this)
    // value 為回調(diào)函數(shù)執(zhí)行的結(jié)果
    let value
    const vm = this.vm
    try {
      // 執(zhí)行回調(diào)函數(shù)孵坚,比如 updateComponent粮宛,進入 patch 階段
      value = this.getter.call(vm, vm)
    } catch (e) {
      if (this.user) {
        handleError(e, vm, `getter for watcher "${this.expression}"`)
      } else {
        throw e
      }
    } finally {
      // "touch" every property so they are all tracked as
      // dependencies for deep watching
      if (this.deep) {
        traverse(value)
      }
      // 關(guān)閉 Dep.target,Dep.target = null
      popTarget()
      this.cleanupDeps()
    }
    return value
  }

總結(jié):

Vue的一部更新機制如何實現(xiàn)的卖宠?

  • vue的異步更新機制是利用瀏覽器的異步任務(wù)隊列實現(xiàn)的(首選事微任務(wù)其次事宏任務(wù))巍杈。
    當(dāng)響應(yīng)式數(shù)據(jù)更新時會調(diào)用dep.notify,通知dep中收集的watcher去執(zhí)行update方法扛伍,watcher.update將watcher放入一個watcher隊列中筷畦。

Vue 的 nextTick API 是如何實現(xiàn)的?

  • vue中的nextTick方法其實做了兩件事:
    1.遞歸回調(diào)函數(shù)用try catch 包裹然后放入到callbacks數(shù)組中。
    2.執(zhí)行timerFun方法鳖宾,在瀏覽器的異步執(zhí)行隊列中加入刷新的callbacks函數(shù)吼砂。
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市鼎文,隨后出現(xiàn)的幾起案子渔肩,更是在濱河造成了極大的恐慌,老刑警劉巖拇惋,帶你破解...
    沈念sama閱讀 206,311評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件周偎,死亡現(xiàn)場離奇詭異,居然都是意外死亡撑帖,警方通過查閱死者的電腦和手機蓉坎,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評論 2 382
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來磷仰,“玉大人袍嬉,你說我怎么就攤上這事≡钇剑” “怎么了伺通?”我有些...
    開封第一講書人閱讀 152,671評論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀的道長逢享。 經(jīng)常有香客問我罐监,道長,這世上最難降的妖魔是什么瞒爬? 我笑而不...
    開封第一講書人閱讀 55,252評論 1 279
  • 正文 為了忘掉前任弓柱,我火速辦了婚禮,結(jié)果婚禮上侧但,老公的妹妹穿的比我還像新娘矢空。我一直安慰自己,他們只是感情好禀横,可當(dāng)我...
    茶點故事閱讀 64,253評論 5 371
  • 文/花漫 我一把揭開白布屁药。 她就那樣靜靜地躺著,像睡著了一般柏锄。 火紅的嫁衣襯著肌膚如雪酿箭。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,031評論 1 285
  • 那天趾娃,我揣著相機與錄音缭嫡,去河邊找鬼。 笑死抬闷,一個胖子當(dāng)著我的面吹牛妇蛀,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 38,340評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼讥耗,長吁一口氣:“原來是場噩夢啊……” “哼有勾!你這毒婦竟也來了疹启?” 一聲冷哼從身側(cè)響起古程,我...
    開封第一講書人閱讀 36,973評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎喊崖,沒想到半個月后挣磨,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,466評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡荤懂,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,937評論 2 323
  • 正文 我和宋清朗相戀三年茁裙,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片节仿。...
    茶點故事閱讀 38,039評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡晤锥,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出廊宪,到底是詐尸還是另有隱情矾瘾,我是刑警寧澤,帶...
    沈念sama閱讀 33,701評論 4 323
  • 正文 年R本政府宣布箭启,位于F島的核電站壕翩,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏傅寡。R本人自食惡果不足惜放妈,卻給世界環(huán)境...
    茶點故事閱讀 39,254評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望荐操。 院中可真熱鬧芜抒,春花似錦、人聲如沸托启。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,259評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽驾中。三九已至唉堪,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間肩民,已是汗流浹背唠亚。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留持痰,地道東北人灶搜。 一個月前我還...
    沈念sama閱讀 45,497評論 2 354
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親割卖。 傳聞我的和親對象是個殘疾皇子前酿,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,786評論 2 345

推薦閱讀更多精彩內(nèi)容