摘要:事件對內(nèi)存和性能的影響雖說事件處理程序可以為現(xiàn)代頁面添加很強的交互能力,但是不分青紅皂白就添加大量的事件處理程序絕對是一種愚蠢的行為。最適合采用事件委托的事件包括和。提交提交某個表單的操作代碼移除事件處理程序提交中。。。
JavaScript 事件對內(nèi)存和性能的影響
雖說事件處理程序可以為現(xiàn)代 Web 頁面添加很強的交互能力,但是不分青紅皂白就添加大量的事件處理程序絕對是一種愚蠢的行為。
我們來分析一下:事件處理程序本質(zhì)上是一種函數(shù),是一種對象,存放在內(nèi)存中,設(shè)置大量的事件處理程序會使內(nèi)存中的對象變多,Web 程序的性能會變得越來越差,用戶體驗很不好。
為了更好地利用好事件處理程序,便出現(xiàn)了事件委托,用來提升性能。
事件委托事件委托(event delegation):把若干個子節(jié)點上的相同事件的處理函數(shù)綁定到它的父節(jié)點上去,在父節(jié)點上統(tǒng)一處理從子節(jié)點冒泡上來的事件,這種技術(shù)就叫做事件委托。
補充一下:事件委托并不局限于父節(jié)點與子節(jié)點之間。也可以這樣玩,比如頁面文檔中有好多個處在不同位置地 button,都是綁定 click 事件,使用事件委托,我們可以把這些個事件統(tǒng)一綁定到 body 元素,然后再進行處理(雖然一般很少這么用)。
下面舉例子逐步說明事件委托的優(yōu)勢:
假設(shè)有上面的代碼,我們現(xiàn)在有一個需求:就是無論單擊上面的列表(ul)的哪個子列表(li),都會彈出一個框,來顯示我們點擊了哪個子列表。
需求不難吧?有了需求,接下來是該寫 js 代碼了,現(xiàn)在有兩種方法放在你眼前:1. 為每個 li 子元素綁定 click 事件,然后設(shè)置處理函數(shù); 2. 利用事件委托,為 ul 父元素綁定 click 事件,然后設(shè)置處理函數(shù)
// 方法一 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);
看著上面的代碼,我這里寫幾點方法二的優(yōu)點:1. 減少了訪問 DOM 的次數(shù),提升了性能;2. 將子元素的事件處理程序統(tǒng)一綁定到其父元素,減少了對內(nèi)存的占用;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("學(xué)的越多,越覺得自己無知!"); break; case "list-2": alert("愛是一種藝術(shù)!"); 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。
事件委托只是一種非常不錯的事件綁定的思想,所以不應(yīng)該拘泥于上面的例子,要活學(xué)活用! ^_^
移除事件處理程序我們前面說過,事件處理程序存在于內(nèi)存中,每當(dāng)將事件處理程序指定給元素時,運行中的瀏覽器代碼與支持頁面交互的 JavaScript 代碼之間就會建立一個連接。這種連接越多,頁面執(zhí)行就越慢。前面所說的事件委托就是用來限制建立的連接數(shù)量。
還有,就是內(nèi)存中那些使用完后不再使用的事件處理程序,如果不釋放掉,也會影響 Web 應(yīng)用程序的內(nèi)存和性能。
var button = document.getElementById("button"); button.onclick = function(){ // 提交某個表單的操作代碼 button.onclick = null; // 移除事件處理程序 event.target.firstChild.nodeValue = "提交中。。。"; };
總的原則就是:移除掉那些過時不再使用的事件處理程序,釋放內(nèi)存!
參考資料【書】《JavaScript 高級程序設(shè)計(第三版)》
【文章】2016前端面試題之手寫事件模型及事件代理/委托(推薦)
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://specialneedsforspecialkids.com/yun/80571.html
摘要:定時刪除在設(shè)置鍵的過期時間的同時,創(chuàng)建一個,讓定時器在鍵的過期時間到達時,立即執(zhí)行對鍵的刪除操作。主動刪除對內(nèi)存友好,但是對時間不友好,有較多過期鍵的而情況下,刪除過期鍵會占用相當(dāng)一部分時間。 1.定時刪除:在設(shè)置鍵的過期時間的同時,創(chuàng)建一個timer,讓定時器在鍵的過期時間到達時,立即執(zhí)行對鍵的刪除操作。(主動刪除)對內(nèi)存友好,但是對cpu時間不友好,有較多過期鍵的而情況下,刪除過期...
摘要:是開發(fā)的一個測試性能的擴展,本文記錄了在應(yīng)用中使用對進行性能優(yōu)化,查找性能瓶頸的方法。函數(shù)用于停止性能分析,并返回分析的數(shù)據(jù)。該參數(shù)用于為剖析結(jié)果添加額外的信息,該參數(shù)的值使用以下宏,如果需要提供多個值,使用進行分隔。 XHProf是facebook 開發(fā)的一個測試php性能的擴展,本文記錄了在PHP應(yīng)用中使用XHProf對PHP進行性能優(yōu)化,查找性能瓶頸的方法。 安裝Xhprof擴展...
閱讀 3044·2021-11-22 09:34
閱讀 3636·2021-08-31 09:45
閱讀 3836·2019-08-30 13:57
閱讀 1670·2019-08-29 15:11
閱讀 1681·2019-08-28 18:04
閱讀 3218·2019-08-28 17:59
閱讀 1558·2019-08-26 13:35
閱讀 2188·2019-08-26 10:12