JavaScript事件委托(事件代理)舉例詳解
是什么
利用 JS 事件冒泡動態(tài)為元素綁定事件的方法稱為事件委托(Event Delegation,也稱為“事件代理”)。
事件委托就是把原本需要綁定在子元素上的事件(onclick、onkeydown 等)委托給它的父元素,讓父元素來監(jiān)聽子元素的冒泡事件,并在子元素發(fā)生事件冒泡時(shí)找到這個(gè)子元素。
為什么
在 JavaScript 中,頁面內(nèi)事件處理程序的個(gè)數(shù)會直接影響頁面的整體性能,因?yàn)槊總€(gè)事件處理程序都是對象,對象會占用內(nèi)存,內(nèi)存中的對象越多,頁面的性能則越差。此外,事件處理程序需要與 DOM 節(jié)點(diǎn)進(jìn)行交互,訪問 DOM 的次數(shù)越多,引起瀏覽器重繪和重排的次數(shù)也就越多,從而影響頁面的性能。
- 重繪是指當(dāng)元素樣式改變時(shí),瀏覽器會根據(jù)元素的新樣式重新繪制元素的外觀。
- 重排是指當(dāng) DOM 樹的一部分發(fā)生變化時(shí)(例如元素尺寸改變),瀏覽器會重新創(chuàng)建 DOM 樹。
事件委托原理回顧
事件委托是利用事件的冒泡原理來實(shí)現(xiàn)的,大致可以分為三個(gè)步驟:
- 確定要添加事件元素的父級元素
- 給父元素定義事件,監(jiān)聽子元素的冒泡事件
- 使用 event.target 來定位觸發(fā)事件冒泡的子元素。
e.target指向的是觸發(fā)事件的元素; e.currentTarget指向的是添加監(jiān)聽事件的元素;
注意:使用事件委托時(shí),并不是說把事件委托給隨意一個(gè)父元素就行。因?yàn)槭录芭莸倪^程也需要消耗時(shí)間,距離越遠(yuǎn),所需的時(shí)間也就越長,所有最好在直接父元素上使用事件委托。
舉例子
1.假如我們要為 ul 列表下的每個(gè) li 標(biāo)簽添加點(diǎn)擊事件,如果不使用事件委托,最簡單的辦法就是使用循環(huán)來為每個(gè) li 標(biāo)簽綁定事件,示例代碼如下:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>JavaScript</title> </head> <body> <ul id="list"> <li>1</li> <li>2</li> <li>3</li> <li>4</li> </ul> <script> window.onload = function(){ var the_ul = document.getElementById('list'); var the_li = the_ul.getElementsByTagName('li'); for( var i=0; i < the_li.length; i++ ){ the_li[i].onclick = function(){ console.log(this.innerHTML) } } } </script> </body> </html>
2.通過上面的代碼可以看出,要為每個(gè) li 標(biāo)簽綁定點(diǎn)擊事件,首先需要找到 ul 標(biāo)簽,然后通過 ul 標(biāo)簽找到所有 li 標(biāo)簽, 最后在通過遍歷所有 li 標(biāo)簽來綁定事件。若使用事件委托的話,就會簡單很多,示例代碼如下:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>JavaScript</title> </head> <body> <ul id="list"> <li>1</li> <li>2</li> <li>3</li> <li>4</li> </ul> <script> window.onload = function(){ var the_ul = document.getElementById('list'); the_ul.onclick = function(e){ console.log(e.target.innerHTML) } } </script> </body> </html>
通過代碼可以看出,使用事件委托我們只需要為 ul 標(biāo)簽綁定事件,當(dāng) li 標(biāo)簽被點(diǎn)擊時(shí),由于事件冒泡的特性,會觸發(fā) ul 標(biāo)簽上的事件,我們只需要在事件中通過 event 對象中的 target 屬性來找到被點(diǎn)擊的 li 標(biāo)簽即可。不過這樣做也有一個(gè)弊端,那就是當(dāng)我們點(diǎn)擊 ul 標(biāo)簽時(shí),也會觸發(fā)事件。
3.另外,如果我們需要?jiǎng)討B(tài)的向 ul 標(biāo)簽中添加 li 標(biāo)簽,同時(shí)也需要在新添加的 li 標(biāo)簽中添加點(diǎn)擊事件,就必須通過事件委托來實(shí)現(xiàn)了,示例代碼如下:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>JavaScript</title> </head> <body> <ul id="list" style="width: 100px;margin:0;float: left;"> <li>1</li> <li>2</li> <li>3</li> <li>4</li> </ul> <button style="float:left;" id="addli">添加一個(gè) li</button> <button style="float:left;" id="delli">刪除一個(gè) li</button> <script> window.onload = function(){ var the_ul = document.getElementById('list'); var the_li = the_ul.getElementsByTagName('li'); var sum = the_li.length the_ul.onclick = function(e){ console.log(e.target.innerHTML) }; document.getElementById('addli').onclick = function (){ var newli = document.createElement("li"); newli.innerHTML = ++sum; the_ul.appendChild(newli); }; document.getElementById('delli').onclick = function (){ the_ul.firstElementChild.remove(); }; } </script> </body> </html>
事件委托的優(yōu)點(diǎn)
1) 減小內(nèi)存消耗使用事件委托可以大量節(jié)省內(nèi)存,減少事件的定義,通過上面的示例可以看出,要為 ul 標(biāo)簽下的所有 li 標(biāo)簽添加點(diǎn)擊事件,如果分別為每個(gè) li 標(biāo)簽綁定事件,不僅寫起來比較繁瑣,而且對內(nèi)存的消耗也非常大。而使用事件委托的方式將點(diǎn)擊事件綁定到 ul 標(biāo)簽上,就可以實(shí)現(xiàn)監(jiān)聽所有 li 標(biāo)簽,簡潔、高效。
2) 動態(tài)綁定事件在網(wǎng)頁中,有時(shí)我們需要?jiǎng)討B(tài)增加或移除頁面中的元素,比如上面示例中動態(tài)的在 ul 標(biāo)簽中添加 li 標(biāo)簽,如果不使用事件委托,則需要手動為新增的元素綁定事件,同時(shí)為刪除的元素解綁事件。而使用事件委托就沒有這么麻煩了,無論是增加還是減少 ul 標(biāo)簽中的 li 標(biāo)簽,即不需要再為新增的元素綁定事件,也不需要為刪除的元素解綁事件。
所以使用事件委托動態(tài)綁定事件可以減少很多重復(fù)工作的。
總結(jié)
要使用事件委托,需要保證事件能夠發(fā)生冒泡,適合使用事件委托的事件有 click、mousedown、mouseup、keydown、keyup、keypress 等。需要注意的是,雖然 mouseover 和 mouseout 事件也會發(fā)生事件冒泡,但處理起來非常麻煩,所以不推薦在 mouseover 和 mouseout 事件中使用事件委托。
另外,對于不會發(fā)生事件冒泡的事件(例如 load、unload、abort、focus、blur 等),則無法使用事件委托。
(參考于:http://c.biancheng.net/view/9380.html)
到此這篇關(guān)于JavaScript事件委托(事件代理)的文章就介紹到這了,更多相關(guān)js事件委托(事件代理)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
微信小程序中使用自定義字體的實(shí)現(xiàn)與體驗(yàn)優(yōu)化
由于微信支持的字體非常有限,不能滿足個(gè)性化的需求,因此在開發(fā)的過程中可能會需要使用自定義字體,下面這篇文章主要給大家介紹了關(guān)于微信小程序中使用自定義字體的實(shí)現(xiàn)與體驗(yàn)優(yōu)化的相關(guān)資料,需要的朋友可以參考下2022-02-02JS監(jiān)聽和響應(yīng)DOM元素的變化的方法
在前端開發(fā)中,處理動態(tài)變化的 DOM(文檔對象模型)很是常見的需求,比如自動化測試中,可能需要監(jiān)控 DOM 變化來驗(yàn)證測試條件,在用戶填寫表單時(shí),某些字段需要即時(shí)驗(yàn)證等,所以本文給大家介紹了JS監(jiān)聽和響應(yīng)DOM元素的變化的方法,需要的朋友可以參考下2024-09-09uniapp小程序自定義tabbar以及初次加載閃屏解決方法
Uniapp小程序可以通過自定義tabbar來實(shí)現(xiàn)更加個(gè)性化的界面設(shè)計(jì),下面這篇文章主要給大家介紹了關(guān)于uniapp小程序自定義tabbar以及初次加載閃屏解決方法,文中通過圖文介紹的非常詳細(xì),需要的朋友可以參考下2023-05-05比較詳細(xì)的關(guān)于javascript中void(0)的具體含義解釋
比較詳細(xì)的關(guān)于javascript中void(0)的具體含義解釋...2007-08-08Vue項(xiàng)目vscode 安裝eslint插件的方法(代碼自動修復(fù))
這篇文章主要介紹了Vue項(xiàng)目vscode 安裝eslint插件的方法 代碼自動修復(fù),需要的朋友可以參考下2020-04-04javascript記錄文本框內(nèi)文字個(gè)數(shù)檢測文字個(gè)數(shù)變化
要對文本框中用戶輸入的文字進(jìn)行記數(shù),在下面顯示出來,經(jīng)研究找到個(gè)不錯(cuò)的方法,可以完美解決在刪除文字后,字?jǐn)?shù)變化問題2014-10-10JavaScript實(shí)現(xiàn)字符串與日期的互相轉(zhuǎn)換及日期的格式化
這篇文章主要介紹了JavaScript實(shí)現(xiàn)字符串與日期的互相轉(zhuǎn)換及日期的格式化的方法,這里格式化使用的是正則表達(dá)式來替換日期后進(jìn)行格式化,需要的朋友可以參考下2016-03-03