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

資訊專欄INFORMATION COLUMN

Vue源碼中的nextTick的實現(xiàn)邏輯

Anshiii / 1629人閱讀

摘要:這是因為在運行時出錯,我們不這個錯誤的話,會導(dǎo)致整個程序崩潰掉。如果沒有向中傳入,并且瀏覽器支持的話,我們的返回的將是一個。如果不支持,就降低到用,整體邏輯就是這樣。。

我們知道vue中有一個api。Vue.nextTick( [callback, context] )
他的作用是在下次 DOM 更新循環(huán)結(jié)束之后執(zhí)行延遲回調(diào)。在修改數(shù)據(jù)之后立即使用這個方法,獲取更新后的 DOM。那么這個api是怎么實現(xiàn)的呢?你肯定也有些疑問或者好奇。下面就是我的探索,分享給大家,也歡迎大家到github上和我進行討論哈~~

首先貼一下vue的源碼,然后我們再一步步的分析

/* @flow */
/* globals MessageChannel */

import { noop } from "shared/util"
import { handleError } from "./error"
import { isIOS, isNative } from "./env"

const callbacks = []
let pending = false

function flushCallbacks () {
  pending = false
  const copies = callbacks.slice(0)
  callbacks.length = 0
  for (let i = 0; i < copies.length; i++) {
    copies[i]()
  }
}

// Here we have async deferring wrappers using both microtasks and (macro) tasks.
// In < 2.4 we used microtasks everywhere, but there are some scenarios where
// microtasks have too high a priority and fire in between supposedly
// sequential events (e.g. #4521, #6690) or even between bubbling of the same
// event (#6566). However, using (macro) tasks everywhere also has subtle problems
// when state is changed right before repaint (e.g. #6813, out-in transitions).
// Here we use microtask by default, but expose a way to force (macro) task when
// needed (e.g. in event handlers attached by v-on).
let microTimerFunc
let macroTimerFunc
let useMacroTask = false

// Determine (macro) task defer implementation.
// Technically setImmediate should be the ideal choice, but it"s only available
// in IE. The only polyfill that consistently queues the callback after all DOM
// events triggered in the same loop is by using MessageChannel.
/* istanbul ignore if */
if (typeof setImmediate !== "undefined" && isNative(setImmediate)) {
  macroTimerFunc = () => {
    setImmediate(flushCallbacks)
  }
} else if (typeof MessageChannel !== "undefined" && (
  isNative(MessageChannel) ||
  // PhantomJS
  MessageChannel.toString() === "[object MessageChannelConstructor]"
)) {
  const channel = new MessageChannel()
  const port = channel.port2
  channel.port1.onmessage = flushCallbacks
  macroTimerFunc = () => {
    port.postMessage(1)
  }
} else {
  /* istanbul ignore next */
  macroTimerFunc = () => {
    setTimeout(flushCallbacks, 0)
  }
}

// Determine microtask defer implementation.
/* istanbul ignore next, $flow-disable-line */
if (typeof Promise !== "undefined" && isNative(Promise)) {
  const p = Promise.resolve()
  microTimerFunc = () => {
    p.then(flushCallbacks)
    // in problematic UIWebViews, Promise.then doesn"t completely break, but
    // it can get stuck in a weird state where callbacks are pushed into the
    // microtask queue but the queue isn"t being flushed, until the browser
    // needs to do some other work, e.g. handle a timer. Therefore we can
    // "force" the microtask queue to be flushed by adding an empty timer.
    if (isIOS) setTimeout(noop)
  }
} else {
  // fallback to macro
  microTimerFunc = macroTimerFunc
}

/**
 * Wrap a function so that if any code inside triggers state change,
 * the changes are queued using a (macro) task instead of a microtask.
 */
export function withMacroTask (fn: Function): Function {
  return fn._withTask || (fn._withTask = function () {
    useMacroTask = true
    const res = fn.apply(null, arguments)
    useMacroTask = false
    return res
  })
}

export function nextTick (cb?: Function, ctx?: Object) {
  let _resolve
  callbacks.push(() => {
    if (cb) {
      try {
        cb.call(ctx)
      } catch (e) {
        handleError(e, ctx, "nextTick")
      }
    } else if (_resolve) {
      _resolve(ctx)
    }
  })
  if (!pending) {
    pending = true
    if (useMacroTask) {
      macroTimerFunc()
    } else {
      microTimerFunc()
    }
  }
  // $flow-disable-line
  if (!cb && typeof Promise !== "undefined") {
    return new Promise(resolve => {
      _resolve = resolve
    })
  }
}

這么多代碼,可能猛的一看,可能有點懵,不要緊,我們一步一步抽出枝干。首先我們看一下這個js文件里的nextTick的定義

export function nextTick (cb?: Function, ctx?: Object) {
  let _resolve
  callbacks.push(() => {
    if (cb) {
      try {
        cb.call(ctx)
      } catch (e) {
        handleError(e, ctx, "nextTick")
      }
    } else if (_resolve) {
      _resolve(ctx)
    }
  })
  if (!pending) {
    pending = true
    if (useMacroTask) {
      macroTimerFunc()
    } else {
      microTimerFunc()
    }
  }
  // $flow-disable-line
  if (!cb && typeof Promise !== "undefined") {
    return new Promise(resolve => {
      _resolve = resolve
    })
  }
}

我將代碼精簡一下。如下所示,下面的代碼估計就比較容易看懂了。把cb函數(shù)放到會掉隊列里去,如果支持macroTask,則利用macroTask在下一個事件循環(huán)中執(zhí)行這些異步的任務(wù),如果不支持macroTask,那就利用microTask在下一個事件循環(huán)中執(zhí)行這些異步任務(wù)。

export function nextTick (cb?: Function, ctx?: Object) {
  callbacks.push(cb)
    if (useMacroTask) {
      macroTimerFunc()
    } else {
      microTimerFunc()
    }
}

這里再次普及一下js的event loop的相關(guān)知識,js中的兩個任務(wù)隊列 :macrotasks、microtasks

macrotasks: script(一個js文件),setTimeout, setInterval, setImmediate, I/O, UI rendering
microtasks: process.nextTick, Promises, Object.observe, MutationObserver

執(zhí)行過程:
1.js引擎從macrotask隊列中取一個任務(wù)執(zhí)行
2.然后將microtask隊列中的所有任務(wù)依次執(zhí)行完
3.再次從macrotask隊列中取一個任務(wù)執(zhí)行
4.然后再次將microtask隊列中所有任務(wù)依次執(zhí)行完
……
循環(huán)往復(fù)

那么我們再看我們精簡掉的代碼都是干什么的呢?我們往異步隊列里放回調(diào)函數(shù)的時候,我們并不是直接放回調(diào)函數(shù),而是包裝了一個函數(shù),在這個函數(shù)里調(diào)用cb,并且用try catch包裹了一下。這是因為cb在運行時出錯,我們不try catch這個錯誤的話,會導(dǎo)致整個程序崩潰掉。 我們還精簡掉了如下代碼

  if (!cb && typeof Promise !== "undefined") {
    return new Promise(resolve => {
      _resolve = resolve
    })
  }

這段代碼是干嘛的呢?也就是說我們用nextTick的時候,還可以有promise的寫法。如果沒有向nextTick中傳入cb,并且瀏覽器支持Promise的話,我們的nextTick返回的將是一個Promise。所以,nextTick的寫法也可以是如下這樣的

 nextTick().then(()=>{console.log("XXXXX")})

vue源碼里關(guān)于nextTick的封裝的思路,也給我們一些非常有益的啟示,就是我們平時在封裝函數(shù)的時候,要想同時指出回調(diào)和promise的話,就可以借鑒vue中的思路。

大致的思路我們已經(jīng)捋順了。但是為什么執(zhí)行macroTimerFunc或者microTimerFunc就會在下一個tick執(zhí)行我們的回調(diào)隊列呢?下面我們來分析一下這兩個函數(shù)的定義。首先我們分析macroTimerFunc

let macroTimerFunc

if (typeof setImmediate !== "undefined" && isNative(setImmediate)) {
  macroTimerFunc = () => {
    setImmediate(flushCallbacks)
  }
} else if (typeof MessageChannel !== "undefined" && (
  isNative(MessageChannel) ||
  // PhantomJS
  MessageChannel.toString() === "[object MessageChannelConstructor]"
)) {
  const channel = new MessageChannel()
  const port = channel.port2
  channel.port1.onmessage = flushCallbacks
  macroTimerFunc = () => {
    port.postMessage(1)
  }
} else {
  /* istanbul ignore next */
  macroTimerFunc = () => {
    setTimeout(flushCallbacks, 0)
  }
}

從上邊的代碼可以看出,如果瀏覽器支持setImmediate,我們就用setImmediate,如果瀏覽器支持MessageChannel,我們就用MessageChannel的異步特性,如果兩者都不支持,我們就降價到setTimeout
,用setTimeout來把callbacks中的任務(wù)在下一個tick中執(zhí)行

  macroTimerFunc = () => {
    setTimeout(flushCallbacks, 0)
  }

分析完macroTimerFunc,下面我們開始分析microTimerFunc,我把vue源碼中關(guān)于microTimerFunc的定義稍微精簡一下

let microTimerFunc;
if (typeof Promise !== "undefined" && isNative(Promise)) {
  const p = Promise.resolve()
  microTimerFunc = () => {
    p.then(flushCallbacks)
  }
} else {
  // fallback to macro
  microTimerFunc = macroTimerFunc
}

從上邊精簡之后的代碼,我們可以看到microTimerFunc的實現(xiàn)思路。如果支持瀏覽器支持promise,就用promise實現(xiàn)。如果不支持,就降低到用macroTimerFunc

over,整體邏輯就是這樣。??粗鴩樔耍_了之后好好分析一下,還是挺簡單的。

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

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

相關(guān)文章

  • 詳細(xì)分析Vue.nextTick()實現(xiàn)

    摘要:因為平時使用都是傳回調(diào)的,所以很好奇什么情況下會為,去翻看官方文檔發(fā)現(xiàn)起新增如果沒有提供回調(diào)且在支持的環(huán)境中,則返回一個。這就對了,函數(shù)體內(nèi)最后的判斷很明顯就是這個意思沒有回調(diào)支持。 Firstly, this paper is based on Vue 2.6.8剛開始接觸Vue的時候,哇nextTick好強,咋就在這里面寫就是dom更新之后,當(dāng)時連什么macrotask、micro...

    DevYK 評論0 收藏0
  • 你不知道$nextTick

    摘要:復(fù)制代碼然后在這個文件里還有一個函數(shù)叫用來把保存的回調(diào)函數(shù)給全執(zhí)行并清空。其實調(diào)用的不僅是開發(fā)者,更新時,也用到了。但是問題又來了,根據(jù)瀏覽器的渲染機制,渲染線程是在微任務(wù)執(zhí)行完成之后運行的。 當(dāng)在代碼中更新了數(shù)據(jù),并希望等到對應(yīng)的Dom更新之后,再執(zhí)行一些邏輯。這時,我們就會用到$nextTickfuncion call...

    番茄西紅柿 評論0 收藏2637
  • Vue源碼詳解之nextTick:MutationObserver只是浮云,microtask才是核

    摘要:后來尤雨溪了解到是將回調(diào)放入的隊列。而且瀏覽器內(nèi)部為了更快的響應(yīng)用戶,內(nèi)部可能是有多個的而的的優(yōu)先級可能更高,因此對于尤雨溪采用的,甚至可能已經(jīng)多次執(zhí)行了的,都沒有執(zhí)行的,也就導(dǎo)致了我們更新操 原發(fā)于我的博客。 前一篇文章已經(jīng)詳細(xì)記述了Vue的核心執(zhí)行過程。相當(dāng)于已經(jīng)搞定了主線劇情。后續(xù)的文章都會對其中沒有介紹的細(xì)節(jié)進行展開。 現(xiàn)在我們就來講講其他支線任務(wù):nextTick和micro...

    陳偉 評論0 收藏0
  • VueJS源碼學(xué)習(xí)——元素在插入和移出 dom 時過渡邏輯

    摘要:原文地址項目地址關(guān)于中使用效果,官網(wǎng)上的解釋如下當(dāng)元素插入到樹或者從樹中移除的時候,屬性提供變換的效果,可以使用來定義變化效果,也可以使用來定義首先第一個函數(shù)是將元素插入,函數(shù)實現(xiàn)調(diào)用了實現(xiàn)代碼如下寫的好的代碼就是文檔,從注釋和命名上就 src/transition 原文地址項目地址 關(guān)于 vue 中使用 transition 效果,官網(wǎng)上的解釋如下: With Vue.js’ tra...

    Dogee 評論0 收藏0
  • Vue原理】NextTick - 源碼版 之 獨立自身

    摘要:盡量把所有異步代碼放在一個宏微任務(wù)中,減少消耗加快異步代碼的執(zhí)行。我們知道,如果一個異步代碼就注冊一個宏微任務(wù)的話,那么執(zhí)行完全部異步代碼肯定慢很多避免頻繁地更新。中就算我們一次性修改多次數(shù)據(jù),頁面還是只會更新一次。 寫文章不容易,點個贊唄兄弟專注 Vue 源碼分享,文章分為白話版和 源碼版,白話版助于理解工作原理,源碼版助于了解內(nèi)部詳情,讓我們一起學(xué)習(xí)吧研究基于 Vue版本 【2.5...

    劉東 評論0 收藏0

發(fā)表評論

0條評論

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