JavaScript 事件對記憶體和效能的影響
雖說事件處理程式可以為現代 Web 頁面新增很強的互動能力,但是不分青紅皁白就新增大量的事件處理程式絕對是一種愚蠢的行為。
我們來分析一下:事件處理程式本質上是一種函式,是一種物件,存放在記憶體中,設定大量的事件處理程式會使記憶體中的物件變多,Web 程式的效能會變得越來越差,使用者體驗很不好。
為了更好地利用好事件處理程式,便出現了事件委託,用來提升效能。
事件委託
事件委託(event delegation):把若干個子節點上的相同事件的處理函式繫結到它的父節點上去,在父節點上統一處理從子節點冒泡上來的事件,這種技術就叫做事件委託。
補充一下:事件委託並不侷限於父節點與子節點之間。也可以這樣玩,比如頁面文件中有好多個處在不同位置地 button,都是繫結 click 事件,使用事件委託,我們可以把這些個事件統一繫結到 body 元素,然後再進行處理(雖然一般很少這麼用)。
下面舉例子逐步說明事件委託的優勢:
<ul id="parent-list"> <li id="list-1">List 1</li> <li id="list-2">List 2</li> <li id="list-3">List 3</li> <li id="list-4">List 4</li> <li id="list-5">List 5</li> </ul>
假設有上面的程式碼,我們現在有一個需求:就是無論單擊上面的列表(ul)的哪個子列表(li),都會彈出一個框,來顯示我們點選了哪個子列表。
需求不難吧?有了需求,接下來是該寫 js 程式碼了,現在有兩種方法放在你眼前:1. 為每個 li 子元素繫結 click 事件,然後設定處理函式; 2. 利用事件委託,為 ul 父元素繫結 click 事件,然後設定處理函式
// 方法一 var list1 = document.getElementById("list-1"); list1.addEventListener("click",function(){ alert(this.firstChild.nodeValue); },false); var list2 = document.getElementById("list-2"); list2.addEventListener("click", function() { alert(this.firstChild.nodeValue); }, false); var list3 = document.getElementById("list-3"); list3.addEventListener("click", function() { alert(this.firstChild.nodeValue); }, false); var list4 = document.getElementById("list-4"); list4.addEventListener("click", function() { alert(this.firstChild.nodeValue); }, false); var list5 = document.getElementById("list-5"); list5.addEventListener("click", function() { alert(this.firstChild.nodeValue); }, false);
// 方法二 var parentList = document.getElementById("parent-list"); parentList.addEventListener("click",function(){ var target = event.target; if(target.nodeName.toLowerCase() === "li"){ alert(target.firstChild.nodeValue); } },false);
看著上面的程式碼,我這裡寫幾點方法二的優點:1. 減少了訪問 DOM 的次數,提升了效能;2. 將子元素的事件處理程式統一繫結到其父元素,減少了對記憶體的佔用;3. 可以更好地管理事件處理程式,比如移除對某個事件處理程式的引用
注意:如果對各個子元素的需求不一樣,我們還可以這樣來改寫上面的方法二:
// 方法二 var parentList = document.getElementById("parent-list"); parentList.addEventListener("click",function(){ var target = event.target; if(target.nodeName.toLowerCase() === "li"){ switch(target.id){ case "list-1": alert("學的越多,越覺得自己無知!"); break; case "list-2": alert("愛是一種藝術!"); break; case "list-3": target.innerHTML = "呵呵,我改了啊!"; break; case "list-4": target.style.background = "#aaa"; break; case "list-5": target.style.color = "red"; target.style.fontSize = "2em"; break; default: break; } } },false);
因為事件委託依賴事件冒泡機制,所以,並不是所有的事件都可以進行事件委託。
最適合採用事件委託的事件包括:click、mousedown、mouseup、keydown、keyup 和 keypress。
事件委託只是一種非常不錯的事件繫結的思想,所以不應該拘泥於上面的例子,要活學活用! ^_^
移除事件處理程式
我們前面說過,事件處理程式存在於記憶體中,每當將事件處理程式指定給元素時,執行中的瀏覽器程式碼與支援頁面互動的 JavaScript 程式碼之間就會建立一個連線。這種連線越多,頁面執行就越慢。前面所說的事件委託就是用來限制建立的連線數量。
還有,就是記憶體中那些使用完後不再使用的事件處理程式,如果不釋放掉,也會影響 Web 應用程式的記憶體和效能。
<button id="button">提交</button> var button = document.getElementById("button"); button.onclick = function(){ // 提交某個表單的操作程式碼 button.onclick = null; // 移除事件處理程式 event.target.firstChild.nodeValue = "提交中。。。"; };
總的原則就是:移除掉那些過時不再使用的事件處理程式,釋放記憶體!
參考資料
- 【書】《JavaScript 高階程式設計(第三版)》
- 【文章】2016前端面試題之手寫事件模型及事件代理/委託(推薦)
相關文章
- 記憶體故障對電腦的影響記憶體
- 虛擬記憶體對 OI 的影響記憶體
- db_files對於oracle使用記憶體的影響Oracle記憶體
- 【原創】ARM平臺記憶體和cache對xenomai實時性的影響記憶體AI
- JavaScript對記憶體的使用JavaScript記憶體
- 硬體環境對系統效能的影響
- 分支對程式碼效能的影響和優化優化
- mysql刪除和更新操作對效能的影響MySql
- max_connections引數對mysql初始化記憶體的影響MySql記憶體
- JDBC記憶體管理—varchar2(4000)的影響JDBC記憶體
- onconfig中對CPU 記憶體的利用率影響的引數記憶體
- 軟體的效能設計(一)介面設計對軟體效能的影響 (轉)
- 軟體的效能設計(二) 臨時物件對軟體效能的影響 (轉)物件
- hashCode竟然不是根據物件記憶體地址生成的?還對記憶體洩漏與偏向鎖有影響?物件記憶體
- 影響MySQL效能的硬體因素MySql
- 影響MySQL效能的硬體因MySql
- DDR4記憶體頻率2400和3000的區別 高頻記憶體與低頻記憶體效能效能對比記憶體
- JAVA 異常對於效能的影響Java
- 軟體效能的設計(三)資料型別對軟體效能的影響 (轉)資料型別
- RAID的概念和RAID對於SQL效能的影響AISQL
- 雙下劃線開頭的記憶體引數對Oracle AMM行為的影響記憶體Oracle
- JavaScript之記憶體溢位和記憶體洩漏JavaScript記憶體溢位
- DB2 HADR對效能的影響DB2
- InnoDB 隔離模式對 MySQL 效能的影響模式MySql
- PCMA:冠狀病毒對事件營銷的影響事件
- c盤記憶體不足會影響絕地求生閃退嗎_c盤記憶體不足影響絕地求生閃退解決方法記憶體
- 【記憶體洩漏和記憶體溢位】JavaScript之深入淺出理解記憶體洩漏和記憶體溢位記憶體溢位JavaScript
- ORM框架和資料庫對系統效能影響的比較ORM框架資料庫
- Linux記憶體效能指標和工具的對應關係圖Linux記憶體指標
- 磁碟排序對Oracle資料庫效能的影響排序Oracle資料庫
- try catch 對程式碼執行的效能影響
- JavaScript的記憶體管理JavaScript記憶體
- 遊戲記憶體對比普通記憶體區別 遊戲記憶體和普通記憶體相差大嗎?遊戲記憶體
- 測試SQLPLUS的ARRAYSIZE對效能的影響SQL
- Arraysize的設定以及對查詢效能的影響
- css屬性的選擇對動畫效能的影響CSS動畫
- JavaScript 中對記憶體的一些瞭解JavaScript記憶體
- 【譯】JavaScript的記憶體管理和 4 種處理記憶體洩漏的方法JavaScript記憶體