簡體   English   中英

MutationObserver 在整個 DOM 中檢測節點的性能

[英]Performance of MutationObserver to detect nodes in entire DOM

我對使用MutationObserver檢測某個 HTML 元素是否添加到 HTML 頁面中的任何位置很感興趣。 例如,我會說我想檢測是否在 DOM 中的任何位置添加了任何<li>

到目前為止,我看到的所有MutationObserver示例都只檢測節點是否添加到特定容器中。 例如:

一些 HTML

<body>

  ...

  <ul id='my-list'></ul>

  ...

</body>

MutationObserver定義

var container = document.querySelector('ul#my-list');

var observer = new MutationObserver(function(mutations){
  // Do something here
});

observer.observe(container, {
  childList: true,
  attributes: true,
  characterData: true,
  subtree: true,
  attributeOldValue: true,
  characterDataOldValue: true
});

所以在這個例子中, MutationObserver被設置為觀察一個非常確定的容器 ( ul#my-list ) 以查看是否有任何<li>附加到它。

如果我想不那么具體,並像這樣在整個 HTML 正文中觀察<li>是否有問題:

var container = document.querySelector('body');

我知道它適用於我為自己設置的基本示例......但是不建議這樣做嗎? 這會導致性能不佳嗎? 如果是這樣,我將如何檢測和衡量該性能問題?

我想可能有一個原因,所有MutationObserver示例都針對其目標容器如此具體……但我不確定。

這個答案主要適用於大而復雜的頁面。

如果在頁面加載/渲染之前附加,未優化的 MutationObserver 回調可以在頁面大而復雜的情況下(例如,5 秒到 7 秒)增加幾秒鍾的頁面加載時間( 1 , 2 )。 回調作為阻止 DOM 進一步處理的微任務執行,並且可以在復雜頁面上每秒觸發數百或數千次。 大多數示例和現有庫都沒有考慮到這種情況,並提供了好看、易於使用但可能會很慢的 JS 代碼。

  1. 始終使用devtools 分析器,並嘗試使您的觀察者回調消耗少於頁面加載期間消耗的總 CPU 時間的 1%。

  2. 避免通過訪問 offsetTop 和類似屬性觸發強制同步布局

  3. 避免使用像 jQuery 這樣復雜的 DOM 框架/庫,更喜歡原生 DOM 的東西

  4. 觀察屬性時,請使用.observe() attributeFilter: ['attr1', 'attr2']選項。

  5. 盡可能以非遞歸方式觀察直接父母( subtree: false )。
    例如,通過遞歸觀察document來等待父元素是有意義的,成功時斷開觀察者,在這個容器元素上附加一個新的非遞歸的。

  6. 當只等待一個具有id屬性的元素時,使用非常快的getElementById而不是枚舉mutations數組(它可能有數千個條目): example

  7. 例如,如果頁面上所需的元素相對較少(例如iframeobject ),請使用getElementsByTagNamegetElementsByClassName返回的實時 HTMLCollection 並重新檢查它們,而不是枚舉超過 100 個元素的mutations

  8. 避免使用querySelector ,尤其是極慢的querySelectorAll

  9. 如果在 MutationObserver 回調中絕對無法避免querySelectorAll ,則首先執行querySelector檢查,如果成功,則繼續querySelectorAll 平均而言,這樣的組合會快得多。

  10. 如果針對 2018 年之前的 Chrome/ium,請不要使用需要回調的內置數組方法,例如 forEach、filter 等,因為在 Chrome 的 V8 中,與經典的for (var i=0 ....)循環(慢 10-100 倍),並且 MutationObserver 回調可能會在復雜的現代頁面上報告數千個節點。

  • 即使在較舊的瀏覽器中,由 lodash 或類似的快速庫支持的替代功能枚舉也可以。
  • 截至 2018 年,Chrome/ium 正在內聯標准數組內置方法。
  1. 如果針對 2019 之前的瀏覽器,請不要在 MutationObserver 回調中使用像for (let v of something)的慢 ES2015 循環for (let v of something)除非您進行編譯以使結果代碼與經典for循環一樣快地運行。

  2. 如果目標是改變頁面的外觀並且您有一種可靠且快速的方法來告訴添加的元素在頁面的可見部分之外,請斷開觀察者的連接並通過setTimeout(fn, 0)安排整個頁面的重新檢查和重新處理:當解析/布局活動的初始爆發完成並且引擎可以“呼吸”甚至可能需要一秒鍾時,它將被執行。 然后,例如,您可以使用 requestAnimationFrame 以塊的形式不顯眼地處理頁面。

  3. 如果處理很復雜和/或需要很多時間,則可能會導致繪制幀很長、無響應/卡頓,因此在這種情況下,您可以使用去抖動或類似技術,例如在外部數組中累積突變並通過以下方式安排運行setTimeout / requestIdleCallback / requestAnimationFrame:

     const queue = []; const mo = new MutationObserver(mutations => { if (!queue.length) requestAnimationFrame(process); queue.push(mutations); }); function process() { for (const mutations of queue) { // .......... } queue.length = 0; }

回到問題:

觀察一個非常確定的容器ul#my-list以查看是否有任何<li>附加到它。

由於li是直接子節點,並且我們尋找添加的節點,因此唯一需要的選項childList: true (請參閱上面的建議 #2)。

new MutationObserver(function(mutations, observer) {
    // Do something here

    // Stop observing if needed:
    observer.disconnect();
}).observe(document.querySelector('ul#my-list'), {childList: true});

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM