Vue中的$nextTick有什麼作用?

来源:https://www.cnblogs.com/smileZAZ/p/18027869
-Advertisement-
Play Games

這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 一、NextTick是什麼 官方對其的定義 在下次 DOM 更新迴圈結束之後執行延遲回調。在修改數據之後立即使用這個方法,獲取更新後的 DOM 什麼意思呢? 我們可以理解成,Vue 在更新 DOM 時是非同步執行的。當數據發生變化,Vue將 ...


這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助

一、NextTick是什麼

官方對其的定義

在下次 DOM 更新迴圈結束之後執行延遲回調。在修改數據之後立即使用這個方法,獲取更新後的 DOM

什麼意思呢?

我們可以理解成,Vue 在更新 DOM 時是非同步執行的。當數據發生變化,Vue將開啟一個非同步更新隊列,視圖需要等隊列中所有數據變化完成之後,再統一進行更新

舉例一下

Html結構

<div id="app"> {{ message }} </div>

構建一個vue實例

const vm = new Vue({
  el: '#app',
  data: {
    message: '原始值'
  }
})

修改message

this.message = '修改後的值1'
this.message = '修改後的值2'
this.message = '修改後的值3'

這時候想獲取頁面最新的DOM節點,卻發現獲取到的是舊值

console.log(vm.$el.textContent) // 原始值

這是因為message數據在發現變化的時候,vue並不會立刻去更新Dom,而是將修改數據的操作放在了一個非同步操作隊列中

如果我們一直修改相同數據,非同步操作隊列還會進行去重

等待同一事件迴圈中的所有數據變化完成之後,會將隊列中的事件拿來進行處理,進行DOM的更新

為什麼要有nexttick

舉個例子

{{num}}
for(let i=0; i<100000; i++){
    num = i
}

如果沒有 nextTick 更新機制,那麼 num 每次更新值都會觸發視圖更新(上面這段代碼也就是會更新10萬次視圖),有了nextTick機制,只需要更新一次,所以nextTick本質是一種優化策略

二、使用場景

如果想要在修改數據後立刻得到更新後的DOM結構,可以使用Vue.nextTick()

第一個參數為:回調函數(可以獲取最近的DOM結構)

第二個參數為:執行函數上下文

// 修改數據
vm.message = '修改後的值'
// DOM 還沒有更新
console.log(vm.$el.textContent) // 原始的值
Vue.nextTick(function () {
  // DOM 更新了
  console.log(vm.$el.textContent) // 修改後的值
})

組件內使用 vm.$nextTick() 實例方法只需要通過this.$nextTick(),並且回調函數中的 this 將自動綁定到當前的 Vue 實例上

this.message = '修改後的值'
console.log(this.$el.textContent) // => '原始的值'
this.$nextTick(function () {
    console.log(this.$el.textContent) // => '修改後的值'
})

$nextTick() 會返回一個 Promise 對象,可以是用async/await完成相同作用的事情

this.message = '修改後的值'
console.log(this.$el.textContent) // => '原始的值'
await this.$nextTick()
console.log(this.$el.textContent) // => '修改後的值'

三、實現原理

源碼位置:/src/core/util/next-tick.js

callbacks也就是非同步操作隊列

callbacks新增回調函數後又執行了timerFunc函數,pending是用來標識同一個時間只能執行一次

export function nextTick(cb?: Function, ctx?: Object) {
  let _resolve;

  // cb 回調函數會經統一處理壓入 callbacks 數組
  callbacks.push(() => {
    if (cb) {
      // 給 cb 回調函數執行加上了 try-catch 錯誤處理
      try {
        cb.call(ctx);
      } catch (e) {
        handleError(e, ctx, 'nextTick');
      }
    } else if (_resolve) {
      _resolve(ctx);
    }
  });

  // 執行非同步延遲函數 timerFunc
  if (!pending) {
    pending = true;
    timerFunc();
  }

  // 當 nextTick 沒有傳入函數參數的時候,返回一個 Promise 化的調用
  if (!cb && typeof Promise !== 'undefined') {
    return new Promise(resolve => {
      _resolve = resolve;
    });
  }
}

timerFunc函數定義,這裡是根據當前環境支持什麼方法則確定調用哪個,分別有:

Promise.thenMutationObserversetImmediatesetTimeout

通過上面任意一種方法,進行降級操作

export let isUsingMicroTask = false
if (typeof Promise !== 'undefined' && isNative(Promise)) {
  //判斷1:是否原生支持Promise
  const p = Promise.resolve()
  timerFunc = () => {
    p.then(flushCallbacks)
    if (isIOS) setTimeout(noop)
  }
  isUsingMicroTask = true
} else if (!isIE && typeof MutationObserver !== 'undefined' && (
  isNative(MutationObserver) ||
  MutationObserver.toString() === '[object MutationObserverConstructor]'
)) {
  //判斷2:是否原生支持MutationObserver
  let counter = 1
  const observer = new MutationObserver(flushCallbacks)
  const textNode = document.createTextNode(String(counter))
  observer.observe(textNode, {
    characterData: true
  })
  timerFunc = () => {
    counter = (counter + 1) % 2
    textNode.data = String(counter)
  }
  isUsingMicroTask = true
} else if (typeof setImmediate !== 'undefined' && isNative(setImmediate)) {
  //判斷3:是否原生支持setImmediate
  timerFunc = () => {
    setImmediate(flushCallbacks)
  }
} else {
  //判斷4:上面都不行,直接用setTimeout
  timerFunc = () => {
    setTimeout(flushCallbacks, 0)
  }
}

無論是微任務還是巨集任務,都會放到flushCallbacks使用

這裡將callbacks裡面的函數複製一份,同時callbacks置空

依次執行callbacks裡面的函數

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

小結:

  1. 把回調函數放入callbacks等待執行
  2. 將執行函數放到微任務或者巨集任務中
  3. 事件迴圈到了微任務或者巨集任務,執行函數依次執行callbacks中的回調

參考文獻

  • https://juejin.cn/post/6844904147804749832

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Linux下MySQL的安裝與使用 安裝前說明 查看是否安裝過MySQL 如果你是用rpm安裝, 檢查一下RPM PACKAGE: rpm -qa | grep -i mysql # -i 忽略大小寫 檢查mysql service: systemctl status mysqld.service ...
  • 大家好,我是 Java陳序員。 最近 Open AI 又火了一把,其新推出的文本生成視頻模型 —— Sora,引起了巨大的關註。 Sora 目前僅僅只是發佈預告視頻,還未開放出具體的 API. 今天,給大家推薦一個最近十分火熱的開源項目,一個支持使用 Sora 模型將文本生成視頻的 Web 客戶端。 ...
  • 前言 我們每天寫vue3項目的時候都會使用setup語法糖,但是你有沒有思考過下麵幾個問題。setup語法糖經過編譯後是什麼樣子的?為什麼在setup頂層定義的變數可以在template中可以直接使用?為什麼import一個組件後就可以直接使用,無需使用components 選項來顯式註冊組件? v ...
  • 寫在前面 本以為可以在家學習一天,結果家裡來了客人拜年,就沒學習上,有點小遺憾吧。 昨天完成從分類管理的前後端代碼複製出文檔管理的前後端代碼,遺留問題是只能選擇一級父分類。值得說的是,昨晚的遺留的問題修複了,開心。 遺留問題 點擊父文檔,彈出警告,從報錯來看那意思就是parent應該是一個對象,我卻 ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 一、是什麼 Server-Side Rendering 我們稱其為SSR,意為服務端渲染 指由服務側完成頁面的 HTML 結構拼接的頁面處理技術,發送到瀏覽器,然後為其綁定狀態與事件,成為完全可交互頁面的過程 先來看看Web3個階段的發展 ...
  • 在當今數字化時代,瀏覽器錄屏技術已經成為了一種強大的工具,用於記錄和分享網頁內容的視覺體驗。無論是用戶體驗測試、教育培訓、產品演示還是遠程協作,瀏覽器錄屏技術都能提供便捷、高效的解決方案。 線上錄屏 | 一個覆蓋廣泛主題工具的高效線上平臺(amd794.com) amd794.com/records ...
  • 我們是袋鼠雲數棧 UED 團隊,致力於打造優秀的一站式數據中台產品。我們始終保持工匠精神,探索前端道路,為社區積累並傳播經驗價值。 本文作者:佳嵐 Suspense Suspense 組件我們並不陌生,中文名可以理解為暫停or懸停 , 在 React16 中我們通常在路由懶載入中配合 Lazy 組件 ...
  • 本文分享自華為雲社區《從HTML到實戰:深入解析BeautifulSoup4的爬蟲奇妙世界》,作者:檸檬味擁抱。 網路上的信息浩如煙海,而爬蟲技術正是幫助我們從中獲取有用信息的重要工具。在爬蟲過程中,解析HTML頁面是一個關鍵步驟,而BeautifulSoup4正是一款功能強大的解析器,能夠輕鬆解析 ...
一周排行
    -Advertisement-
    Play Games
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...
  • 目錄前言PostgreSql安裝測試額外Nuget安裝Person.cs模擬運行Navicate連postgresql解決方案Garnet為什麼要選擇Garnet而不是RedisRedis不再開源Windows版的Redis是由微軟維護的Windows Redis版本老舊,後續可能不再更新Garne ...
  • C#TMS系統代碼-聯表報表學習 領導被裁了之後很快就有人上任了,幾乎是無縫銜接,很難讓我不想到這早就決定好了。我的職責沒有任何變化。感受下來這個系統封裝程度很高,我只要會調用方法就行。這個系統交付之後不會有太多問題,更多應該是做小需求,有大的開發任務應該也是第二期的事,嗯?怎麼感覺我變成運維了?而 ...
  • 我在隨筆《EAV模型(實體-屬性-值)的設計和低代碼的處理方案(1)》中介紹了一些基本的EAV模型設計知識和基於Winform場景下低代碼(或者說無代碼)的一些實現思路,在本篇隨筆中,我們來分析一下這種針對通用業務,且只需定義就能構建業務模塊存儲和界面的解決方案,其中的數據查詢處理的操作。 ...
  • 對某個遠程伺服器啟用和設置NTP服務(Windows系統) 打開註冊表 HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\W32Time\TimeProviders\NtpServer 將 Enabled 的值設置為 1,這將啟用NTP伺服器功 ...
  • title: Django信號與擴展:深入理解與實踐 date: 2024/5/15 22:40:52 updated: 2024/5/15 22:40:52 categories: 後端開發 tags: Django 信號 松耦合 觀察者 擴展 安全 性能 第一部分:Django信號基礎 Djan ...
  • 使用xadmin2遇到的問題&解決 環境配置: 使用的模塊版本: 關聯的包 Django 3.2.15 mysqlclient 2.2.4 xadmin 2.0.1 django-crispy-forms >= 1.6.0 django-import-export >= 0.5.1 django-r ...
  • 今天我打算整點兒不一樣的內容,通過之前學習的TransformerMap和LazyMap鏈,想搞點不一樣的,所以我關註了另外一條鏈DefaultedMap鏈,主要調用鏈為: 調用鏈詳細描述: ObjectInputStream.readObject() DefaultedMap.readObject ...
  • 後端應用級開發者該如何擁抱 AI GC?就是在這樣的一個大的浪潮下,我們的傳統的應用級開發者。我們該如何選擇職業或者是如何去快速轉型,跟上這樣的一個行業的一個浪潮? 0 AI金字塔模型 越往上它的整個難度就是職業機會也好,或者說是整個的這個運作也好,它的難度會越大,然後越往下機會就會越多,所以這是一 ...
  • @Autowired是Spring框架提供的註解,@Resource是Java EE 5規範提供的註解。 @Autowired預設按照類型自動裝配,而@Resource預設按照名稱自動裝配。 @Autowired支持@Qualifier註解來指定裝配哪一個具有相同類型的bean,而@Resourc... ...