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

Vue中nextTick的原理分析

 更新時間:2025年02月21日 16:43:52   作者:遇見很ok  
本文詳細介紹了Vue中nextTick的原理和使用方法,Vue采用異步渲染機制,通過nextTick可以確保在DOM更新后執(zhí)行回調(diào),nextTick依賴微任務(wù)和宏任務(wù),優(yōu)先使用Promise,降級到MutationObserver/setTimeout,Vue3進一步優(yōu)化了nextTick,減少了復(fù)雜度并提升了性能

Vue中nextTick原理

1. 為什么需要 nextTick

Vue 采用 異步渲染機制,當(dāng)響應(yīng)式數(shù)據(jù)發(fā)生變化時,Vue 并不會立即更新 DOM,而是將這些變化放入一個 隊列 中,并在 同一事件循環(huán)(Event Loop)中合并相同的修改,最后執(zhí)行批量更新。

這樣做的目的是 提升性能,避免不必要的重復(fù)渲染。

例如:

<template>
  <div>{{ msg }}</div>
</template>

<script>
export default {
  data() {
    return {
      msg: "Hello"
    };
  },
  mounted() {
    this.msg = "Vue";
    console.log(document.querySelector("div").innerText); // 仍然是 "Hello"
    this.$nextTick(() => {
      console.log(document.querySelector("div").innerText); // 現(xiàn)在是 "Vue"
    });
  }
};
</script>

為什么 console.log 還是 "Hello"?

因為 Vue 在 this.msg = "Vue"不會立即更新 DOM,而是等本輪事件循環(huán)結(jié)束后再更新。因此,我們需要 nextTick 來確保獲取到更新后的 DOM。

2. nextTick 的原理

Vue 的 nextTick 本質(zhì)上是一個 異步任務(wù)調(diào)度器,它會在當(dāng)前 DOM 任務(wù)完成后執(zhí)行回調(diào)。其內(nèi)部原理主要依賴 微任務(wù)(Microtask)和 宏任務(wù)(Macrotask)。

2.1 任務(wù)隊列

Vue 內(nèi)部維護了一個 回調(diào)隊列(callback queue),當(dāng) nextTick 被調(diào)用時,它會將回調(diào)函數(shù) 推入隊列,然后等待 Vue 進行 DOM 更新后,再依次執(zhí)行這些回調(diào)。

2.2 任務(wù)調(diào)度策略

nextTick 采用 優(yōu)雅降級 的策略,在不同環(huán)境下選擇最佳的異步方法:

  1. Promise(Microtask)(首選,現(xiàn)代瀏覽器支持)
  2. MutationObserver(Microtask)(比 setTimeout 更快)
  3. setImmediate(Macrotask)(僅 IE 支持)
  4. setTimeout(Macrotask)(最后的兜底方案)

代碼實現(xiàn):

function nextTick(callback) {
  const p = Promise.resolve();
  p.then(callback);
}

在 Vue 3 中:

let callbacks = [];
let pending = false;

function flushCallbacks() {
  pending = false;
  const copies = callbacks.slice(0);
  callbacks.length = 0;
  for (let cb of copies) {
    cb();
  }
}

export function nextTick(cb) {
  callbacks.push(cb);
  if (!pending) {
    pending = true;
    Promise.resolve().then(flushCallbacks);
  }
}

流程解析:

  1. 每次調(diào)用 nextTick(cb),將 cb 放入 callbacks 隊列中。
  2. 只要 pending === false,就啟動 微任務(wù)(Promise.then)。
  3. 微任務(wù)執(zhí)行 flushCallbacks,依次調(diào)用 callbacks 隊列中的所有回調(diào)。

3. nextTick 在 Vue 2 和 Vue 3 的區(qū)別

3.1 Vue 2 的 nextTick

在 Vue 2 中,nextTick 主要依賴:

  • Microtask(Promise.then, MutationObserver)
  • Macrotask(setImmediate, setTimeout)
  • 維護了一個 異步任務(wù)隊列,用于批量執(zhí)行 nextTick 回調(diào)。

3.2 Vue 3 的 nextTick

Vue 3 主要優(yōu)化:

  • 只使用 Promise 作為微任務(wù)(不再使用 MutationObserver)。
  • 更高效的 異步隊列處理機制。

Vue 3 中的 nextTick

const resolvedPromise = Promise.resolve();
export function nextTick(fn) {
  return fn ? resolvedPromise.then(fn) : resolvedPromise;
}

優(yōu)化點

  • 直接使用 Promise.resolve().then(fn),避免了 Vue 2 復(fù)雜的回調(diào)隊列管理。
  • 如果不傳入 fn,則返回一個 Promise,支持 await this.$nextTick()。

4. nextTick 的使用場景

4.1 在 DOM 更新后執(zhí)行操作

<template>
  <div ref="box">{{ message }}</div>
</template>

<script>
export default {
  data() {
    return { message: "Hello" };
  },
  methods: {
    updateMessage() {
      this.message = "Vue";
      this.$nextTick(() => {
        console.log(this.$refs.box.innerText); // "Vue"
      });
    }
  }
};
</script>

4.2 在 watch 中等待 DOM 更新

watch(() => state.count, async (newVal) => {
  await nextTick();
  console.log(document.querySelector("#counter").innerText); // 確保 DOM 已更新
});

4.3 在 Vue 3 setup 中使用

import { nextTick, ref } from "vue";

setup() {
  const message = ref("Hello");
  
  const updateMessage = async () => {
    message.value = "Vue";
    await nextTick();
    console.log(document.querySelector("#msg").innerText);
  };

  return { message, updateMessage };
}

總結(jié)

  • nextTick 是 Vue 提供的一個 異步任務(wù)調(diào)度方法,用于在 DOM 更新后執(zhí)行回調(diào)。
  • Vue 采用 異步批量更新 機制,nextTick 可確保 數(shù)據(jù)變更后獲取到最新的 DOM
  • Vue 內(nèi)部采用 Promise(Microtask)優(yōu)先,降級到 MutationObserver / setTimeout 作為備用方案。
  • Vue 3 進一步優(yōu)化了 nextTick,減少了不必要的復(fù)雜度,提升了性能。

你可以簡單理解為:

Vue 在修改數(shù)據(jù)后,不會立即更新 DOM,而是 批量合并修改,并在下一次 事件循環(huán)(Event Loop)結(jié)束時更新 DOM。nextTick 讓你可以等到 DOM 更新完成后再執(zhí)行操作。

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • vuedraggable實現(xiàn)拖拽功能

    vuedraggable實現(xiàn)拖拽功能

    這篇文章主要為大家詳細介紹了vuedraggable實現(xiàn)拖拽功能,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-04-04
  • .eslintrc配置目錄及配置項的使用方式

    .eslintrc配置目錄及配置項的使用方式

    這篇文章主要介紹了.eslintrc配置目錄及配置項的使用方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-10-10
  • element?時間選擇器禁用選擇的使用示例

    element?時間選擇器禁用選擇的使用示例

    最近做項目遇到的一個功能,禁止用戶在輸入內(nèi)容的時候選擇今天以前的日期或者包含今日的日期,本文主要介紹了element?時間選擇器禁用選擇的使用示例,感興趣的可以了解一下
    2023-09-09
  • vue?跳轉(zhuǎn)頁面$router.resolve和$router.push案例詳解

    vue?跳轉(zhuǎn)頁面$router.resolve和$router.push案例詳解

    這篇文章主要介紹了vue?跳轉(zhuǎn)頁面$router.resolve和$router.push案例詳解,這樣實現(xiàn)了既跳轉(zhuǎn)了新頁面,又不會讓后端檢測到頁面鏈接不安全之類的,需要的朋友可以參考下
    2023-10-10
  • vue指令防止按鈕連點解析

    vue指令防止按鈕連點解析

    這篇文章主要介紹了vue指令防止按鈕連點解析,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-04-04
  • vue+axios+mock.js環(huán)境搭建的方法步驟

    vue+axios+mock.js環(huán)境搭建的方法步驟

    本篇文章主要介紹了vue+axios+mock.js環(huán)境搭建的方法步驟,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-08-08
  • vue實現(xiàn)彈出框內(nèi)嵌頁面展示并添加tab切換展示實時加載

    vue實現(xiàn)彈出框內(nèi)嵌頁面展示并添加tab切換展示實時加載

    彈窗效果是在Web開發(fā)中經(jīng)常用到的一種交互效果,這篇文章主要給大家介紹了關(guān)于vue實現(xiàn)彈出框內(nèi)嵌頁面展示并添加tab切換展示實時加載的相關(guān)資料,需要的朋友可以參考下
    2024-01-01
  • vue3引入Element-plus的詳細步驟記錄

    vue3引入Element-plus的詳細步驟記錄

    Element Plus是為適配Vue3而對Element UI進行重構(gòu)后產(chǎn)生的前端組件庫,包含豐富的基礎(chǔ)組件,下面這篇文章主要給大家介紹了關(guān)于vue3引入Element-plus的相關(guān)資料,需要的朋友可以參考下
    2022-04-04
  • vue項目中使用百度地圖的方法

    vue項目中使用百度地圖的方法

    這篇文章主要介紹了在vue項目中使用百度地圖的方法,本文通過實例代碼給大家介紹的非常詳細,具有一定的參考借鑒借鑒價值,需要的朋友可以參考下
    2018-06-06
  • 測試平臺開發(fā)vue組件化重構(gòu)前端代碼

    測試平臺開發(fā)vue組件化重構(gòu)前端代碼

    這篇文章主要為大家介紹了測試平臺開發(fā)vue組件化重構(gòu)前端代碼,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-05-05

最新評論