JavaScript 性能優化實戰:減少 DOM 操作引發的重排與重繪

在前端開發中,DOM 操作是 JavaScript 性能優化的核心痛點之一。頻繁的 DOM 操作會觸發瀏覽器的 重排(Reflow)重繪(Repaint),導致性能顯著下降。本文將深入分析這一瓶頸,并通過實際案例展示優化策略。


一、問題定位:DOM 操作的性能代價

1.?重排與重繪的原理

  • 重排(Reflow):瀏覽器重新計算元素的幾何屬性(如位置、大小),并更新渲染樹。
  • 重繪(Repaint):瀏覽器根據渲染樹重新繪制像素到屏幕。
  • 觸發條件
    • 修改 DOM 結構(新增/刪除節點)
    • 改變樣式(如?widthheightmargin
    • 獲取布局屬性(如?offsetWidthgetComputedStyle

2.?性能瓶頸示例

// ? 低效寫法:循環中頻繁操作 DOM
const list = document.getElementById('list');
for (let i = 0; i < 100; i++) {const item = document.createElement('li');item.textContent = `Item ${i}`;list.appendChild(item); // 每次觸發重排
}

問題:每次 appendChild 會強制瀏覽器進行一次重排,100 次循環會導致 100 次重排,顯著降低性能。


二、優化策略:批量操作 DOM

1.?使用 DocumentFragment

通過 DocumentFragment 在內存中構建 DOM 樹,一次性插入頁面,減少重排次數。

// ? 優化方案:使用 DocumentFragment
const fragment = document.createDocumentFragment();
for (let i = 0; i < 100; i++) {const item = document.createElement('li');item.textContent = `Item ${i}`;fragment.appendChild(item);
}
document.getElementById('list').appendChild(fragment); // 單次重排

2.?效果對比

方法重排次數首屏渲染時間
原始寫法(循環插入)100 次~800ms
使用 DocumentFragment1 次~150ms

三、真實案例:電商后臺商品列表優化

1.?背景

某電商平臺后臺需要動態加載 10,000 條商品記錄,原始代碼直接通過循環創建 DOM 節點,導致頁面卡頓,首屏加載時間超過 2 秒。

2.?優化方案

  • 虛擬滾動(Virtual Scrolling):僅渲染可視區域內的節點(如 50 條),通過滾動事件動態更新內容。
  • 批量 DOM 操作:使用?DocumentFragment?或?requestAnimationFrame?合并操作。
// 虛擬滾動示例(簡化版)
const visibleItems = 50;
const container = document.getElementById('list');
const fragment = document.createDocumentFragment();for (let i = 0; i < visibleItems; i++) {const item = document.createElement('div');item.textContent = `Product ${i}`;fragment.appendChild(item);
}container.appendChild(fragment); // 首屏加載 50 條,后續按需加載

3.?優化結果

  • 內存占用下降 60%
  • 首屏加載時間從 1.2s 縮短至 0.15s
  • 滾動流暢度提升 3 倍(FPS 從 15 提升至 45)

四、最佳實踐總結

  1. 避免在循環中操作 DOM
    將 DOM 操作集中到內存中完成(如 DocumentFragment),最后一次性插入。

  2. 優先使用 requestAnimationFrame
    對于動畫或高頻事件(如 scroll),通過 requestAnimationFrame 合并任務,減少重排觸發頻率。

  3. 工具輔助分析
    使用 Chrome DevTools 的 Performance 面板 監控重排/重繪頻率,定位性能瓶頸。

  4. 虛擬化大數據渲染
    對千級以上的數據渲染場景,采用虛擬滾動或分頁策略,減少 DOM 節點數量。


五、結語

DOM 操作引發的重排與重繪是 JavaScript 性能優化中的關鍵問題。通過 批量操作虛擬化渲染,可以顯著減少瀏覽器的計算負擔,提升用戶體驗。在實際項目中,建議結合 Chrome DevTools 分析性能瓶頸,并針對性地實施優化策略。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/web/87323.shtml
繁體地址,請注明出處:http://hk.pswp.cn/web/87323.shtml
英文地址,請注明出處:http://en.pswp.cn/web/87323.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

力扣 hot100 Day33

24. 兩兩交換鏈表中的節點 給你一個鏈表&#xff0c;兩兩交換其中相鄰的節點&#xff0c;并返回交換后鏈表的頭節點。你必須在不修改節點內部的值的情況下完成本題&#xff08;即&#xff0c;只能進行節點交換&#xff09;。 //抄的 class Solution { public:ListNode* swapP…

DevExpress V25.1 版本更新,開啟控件AI新時代

WinForms Controls v25.1 AI 驅動的語義搜索 我們的 WinForms 數據網格、GridLookUpEdit 和 SearchLookUpEdit 控件具有增強的搜索體驗&#xff0c;使用戶能夠更快/更準確地在大型數據集中查找相關數據。與基于關鍵字的標準搜索不同&#xff0c;語義搜索利用自然語言處理 &…

【分層圖 虛擬節點】 P11327 [NOISG 2022 Finals] Voting Cities|普及+

本文涉及知識點 C圖論 P11327 [NOISG 2022 Finals] Voting Cities 題目描述 你所在的國家的國家主席 L o r d P o o t y \bf{Lord\ Pooty} Lord Pooty 將要退休了&#xff01;他希望選擇他的一個兒子作為他的繼承人&#xff0c;出于各方面因素的考慮&#xff0c;他決定進行…

Web3云服務商安全性怎么選

Web3安全之錨&#xff1a;為何阿里云是企業級應用的首選? 隨著Web3、去中心化金融&#xff08;DeFi&#xff09;和數字資產的浪潮席卷全球&#xff0c;無數開發者和企業涌入這個充滿機遇的新賽道。然而&#xff0c;機遇背后是同樣巨大的安全挑戰。從智能合約漏洞到大規模DDoS…

uniapp加上全局水印

文章目錄 一、效果圖二、創建watermark.js文件三、在main.js中引入四、運行 前言&#xff1a;uniapp頁面加水印你還在傻乎乎的一個個頁面加嗎&#xff0c;今天教你一招&#xff0c;一步到位 一、效果圖 未登錄效果 登錄后效果 二、創建watermark.js文件 這里的水印因為我…

thinkphp8.0七牛云直傳圖片

環境&#xff1a;tp8\php8.3; 服務器&#xff1a;centOS Stream 9; 場景&#xff1a;通過html頁面直傳七牛云服務器&#xff0c;速度更快&#xff1b; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta na…

Godot x openKylin 全國開發大賽正式啟動

從2023年開始&#xff0c;Godot Hub 每年舉辦一次 Godot Hub Festival 開發大賽&#xff0c;現已成為國內 Godot 社區規模最大的開發比賽。本屆 Godot Hub Festival 2025將與 OpenAtom openKylin 開源社區合作舉辦&#xff0c;定名為 Godot x openKylin 全國開發大賽&#xff0…

工控機Linux修改網口

修改Ip:sudo nmcli connection modify net1-static ipv4.addresses 192.168.200.225/24 修改dns:sudo nmcli connection modify net1-static ipv4.dns 114.114.114.114 修改網關:sudo nmcli connection modify net1-static ipv4.gateway 192.168.200.1 IP生效&#xff1a;nm…

CRMEB Pro版v3.3源碼全開源+PC端+Uniapp前端+搭建教程

一.介紹 crmeb Pro版 v3.3版本正式發布&#xff0c;全新UI重磅上線&#xff0c;煥然一新&#xff0c;不負期待&#xff01;頁面DIY設計功能全面升級&#xff0c;組件更豐富&#xff0c;樣式設計更全面&#xff1b;移動端商家管理&#xff0c;讓商城管理更便捷&#xff0c;還從…

【python】OOP:Object-Oriented Programming

文章目錄 1. 面向對象編程的核心概念1.1 類與對象的關系1.2 封裝&#xff08;Encapsulation&#xff09; 2. 繼承與多態2.1 繼承機制2.2 多重繼承2.3 多態性 3. 特殊方法與運算符重載4. 抽象類與接口4.1 抽象基類 5. 組合與聚合6. 屬性管理6.1 使用property裝飾器6.2 描述符協議…

蒙特卡洛方法:隨機抽樣的藝術與科學

本文由「大千AI助手」原創發布&#xff0c;專注用真話講AI&#xff0c;回歸技術本質。拒絕神話或妖魔化。搜索「大千AI助手」關注我&#xff0c;一起撕掉過度包裝&#xff0c;學習真實的AI技術&#xff01; 蒙特卡洛算法&#xff08;Monte Carlo Method&#xff09;是一類基于隨…

Linux基礎 -- UBI(**Unsorted Block Images**)

UBI&#xff08;Unsorted Block Images&#xff09;是 Linux 中為原始 NAND Flash 設計的一種 邏輯卷管理層&#xff0c;其核心作用是&#xff1a;在 NAND 閃存設備上提供 壞塊管理、擦寫均衡&#xff08;wear leveling&#xff09;和邏輯到物理地址映射等機制&#xff0c;為上…

線程相關函數

思維導圖 1. 創建一個分支線程&#xff0c;在主線程中拷貝文件的前一部分&#xff0c;主線程拷貝后一部分。 2.解讀代碼 info1 from child process_1 info1 from parent process3.解讀代碼&#xff0c;-打印多少次 14次

SeaTunnel 社區月報(5-6 月):全新功能上線、Bug 大掃除、Merge 之星是誰?

在 5 月和 6 月&#xff0c;SeaTunnel 社區迎來了一輪密集更新&#xff1a;2.3.11 正式發布&#xff0c;新增對 Databend、Elasticsearch 向量、HTTP 批量寫入、ClickHouse 多表寫入等多個連接器能力&#xff0c;全面提升了數據同步靈活性。同時&#xff0c;近 100 個修復與優化…

數學建模_非線性規劃

matlab求解調用示例 第二道例題建模 matlab求解 1.matlab只能處理min問題&#xff1a; max兩邊取負號變成min 2. > > >號變成 < < <&#xff1a;兩邊取負號 調用示例 第二道例題建模 目標函數取平方而不取絕對值 后面省略

【BurpSuite 2025最新版插件開發】基礎篇7:數據的持久化存儲

1 前言 歷史章節&#xff1a; 【BurpSuite 2025最新版插件開發】基礎篇1&#xff1a;環境搭建 【BurpSuite 2025最新版插件開發】基礎篇2&#xff1a;插件生命周期與核心接口 【BurpSuite 2025最新版插件開發】基礎篇3&#xff1a;請求攔截和修改簡單示例 【BurpSuite 202…

GPT-4 Turbo集成智能工作流,開啟自動化研究與知識管理新篇章!

目錄 一、系統架構設計二、核心模塊實現1. 智能數據采集引擎2. 自動化研究引擎3. 知識管理系統 三、智能工作流引擎四、關鍵技術實現1. 動態工作流引擎2. 知識圖譜構建 五、企業級部署方案1. 云原生架構2. Docker部署腳本 六、應用案例&#xff1a;藥物研發項目七、性能優化策略…

告別SQL卡頓與混亂!AI如何賦能實時計算?

在當今數據驅動的商業環境中&#xff0c;SQL作為與數據庫交互的核心語言&#xff0c;其編寫效率和質量直接影響著企業的數據決策速度和系統性能。然而&#xff0c;我們在長期的企業服務實踐中發現&#xff0c;數據庫開發人員普遍面臨以下痛點&#xff1a; SQL性能問題頻發&…

LeetCode算法(和中等打的有來有回)——盛最多水的容器

文章目錄 leetcode第11題&#xff1a;盛最多水的容器二次循環代碼 雙指針優化解析代碼 leetcode第11題&#xff1a;盛最多水的容器 二次循環 這道題比較容易想到的就是通過二次循環遍歷所有能盛的水的體積。 代碼 class Solution {public int maxArea(int[] height) {// 記錄…

Karmada 多集群服務發現

一、背景介紹 多集群架構下&#xff0c;不同 Kubernetes 集群間的服務如何互通是核心挑戰。Karmada 支持 Kubernetes Multi?cluster Service APIs&#xff08;MCS&#xff09;&#xff0c;通過 ServiceExport 和 ServiceImport 實現跨集群服務發現與調用&#xff0c;幫助多集…