使用原生JS實(shí)現(xiàn)事件委托

事件

在使用JavaScript與DOM交互時(shí),事件是用到的比較多的.

JavaScript的事件機(jī)制是一個(gè)標(biāo)準(zhǔn)的觀察者模式(Observer Pattern),是一種抽象的訂閱者和開(kāi)發(fā)者的模式.我們可以在節(jié)點(diǎn)上添加上指定條件下的觸發(fā)事件.

事件綁定和事件監(jiān)聽(tīng)

在DOM level 0中事件是直接綁定的,比如

buttom.onclick = function(){}

到了DOM level 2的時(shí)候增加了事件監(jiān)聽(tīng),比如

button.addEventListener('click',function(){})

事件綁定的方法有個(gè)缺點(diǎn)就是它只能綁定一次,而使用addEventListener 可以綁定多個(gè)函數(shù),執(zhí)行順序按照綁定的先后順序來(lái)執(zhí)行.

在使用過(guò)程中發(fā)現(xiàn),如果我們?cè)谛枰脑厣隙继砑恿吮O(jiān)聽(tīng),那這是非常消耗資源和性能的,所以才有了事件委托.

什么是事件委托

事件委托是為了避免我們?cè)谔囟ǖ拿總€(gè)節(jié)點(diǎn)上都添加事件監(jiān)聽(tīng)器,而在父節(jié)點(diǎn)上來(lái)監(jiān)聽(tīng)冒泡上來(lái)的事件.

事件委托是基于冒泡機(jī)制的,所以本篇重點(diǎn)關(guān)注冒泡階段.瀏覽器的

冒泡

冒泡如下圖所示,element2的事件先被觸發(fā),然后是element1

               / \
---------------| |-----------------
| element1     | |                |
|   -----------| |-----------     |
|   |element2  | |          |     |
|   -------------------------     |
|        Event BUBBLING           |
-----------------------------------

舉例說(shuō)明

以下的例子我們都使用addEventListener ,其中參數(shù)e 為瀏覽器傳給我們的一個(gè)包含了事件各種屬性的對(duì)象.

第一個(gè)例子

現(xiàn)在我們有一個(gè)需求,要在點(diǎn)擊列表里面的時(shí)候在控制臺(tái)打印出hello.

先寫(xiě)HTML,稍微加點(diǎn)樣式

<!DOCTYPE html>
<html>
<head>
  <meta charset="utf-8">
  <meta name="viewport" content="width=device-width">
  <title>Test</title>
  <style>
    li{background-color: red;}
    ul{border: 20px solid green;}
  </style>
</head>
<body>
<ul>
  <li>one</li>
  <li>two</li>
  <li>three</li>
  <li>four</li>
</ul>
</body>
</html>

要在點(diǎn)擊li的時(shí)候打出hello,

var liName = document.querySelectorAll('li')
function fn(e){
  if(e.target.tagName === 'LI' ){ //target為當(dāng)前點(diǎn)擊的元素
    console.log('hello') 
    }
}
//在每個(gè)li上添加監(jiān)聽(tīng)
liName[0].addEventListener('click',fn)
liName[1].addEventListener('click',fn)
liName[2].addEventListener('click',fn)
liName[3].addEventListener('click',fn)

每個(gè)元素上都添加監(jiān)聽(tīng)是非常耗資源的,我們可以在li的父元素ul上添加一個(gè)監(jiān)聽(tīng)事件

var ulTest = document.querySelector('ul')
function fn(e){
  if(e.target.tagName === 'LI' ){
    console.log('hello') 
    }
}
//添加ul上的監(jiān)聽(tīng)事件
ulTest.addEventListener('click',fn)

上面的代碼同樣可以達(dá)到同樣的需求,而且占用的資源更少.

那么如果我們需要監(jiān)聽(tīng)的元素不是li,而是li的子元素呢?

......
<ul>
  <li><span>one</span></li>
  <li>two</li>
  <li>three</li>
  <li>four</li>
</ul>
......

改寫(xiě):

var ulTest = document.querySelector('ul')
var bodyTest = document.querySelector('body')
function fn(e){
  var el = e.target  
  while(el.tagName !== 'LI'){
    el = el.parentNode
  }
  if(el){
  console.log('hello')
  }else{
    console.log('111')
  }
}

//添加ul上的監(jiān)聽(tīng)事件
ulTest.addEventListener('click',fn)

上面的代碼中,點(diǎn)擊li中的span,ul也能夠出發(fā)監(jiān)聽(tīng)事件.會(huì)在用戶點(diǎn)擊的元素一級(jí)一級(jí)的往上找(冒泡),找到符合條件的就調(diào)用事件.

第二個(gè)例子

在div中有p和h1,其中又各有一個(gè)span.我們需要在點(diǎn)擊h1或h1里的span的時(shí)候觸發(fā)事件.

<!DOCTYPE html>
<html>
<head>
  <meta charset="utf-8">
  <meta name="viewport" content="width=device-width">
  <title>Test</title>
  <style>
    p{ background-color: red;}
    h1{ background-color: green;}
    span{ background-color: yellow;}
  </style>
</head>
<body>
<div>
  <p>我是<span>p</span></p>
  <h1>我是<span>h1</span></h1>
</div>
</body>
</html>

JS

var div = document.querySelector('div')

div.addEventListener('click',function(e){
  var t = e.target
  //尋找H1節(jié)點(diǎn)
  while(t.tagName !== 'H1'){
    if(t === div){
      t = null
      break;
    }
     t = t.parentNode
  }
  if(t){
    console.log('你找到了h1')
  }else{
    console.log('你點(diǎn)擊了其他元素')
  }
})

在上面的代碼中,首先在最外層的div上添加監(jiān)聽(tīng)事件,在while循環(huán)中我們判斷當(dāng)前的點(diǎn)擊元素是否為H1元素,如果不是我們往上查找,直到查到H1元素.如果查到了最外層的div元素,證明沒(méi)有在H1元素之內(nèi),我們就不需要再往上去查找了,所以這里我們將元素置為null.

參考

http://pij.robinqu.me/Browser_Scripting/DOM_Scripting/EventAPI.html

https://developer.mozilla.org/zh-CN/docs/Web/API/EventTarget/addEventListener

https://www.quirksmode.org/js/events_order.html

https://developer.mozilla.org/zh-CN/docs/Web/API/Event

http://javascript.ruanyifeng.com/dom/event.html

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市铁追,隨后出現(xiàn)的幾起案子革砸,更是在濱河造成了極大的恐慌庄涡,老刑警劉巖面徽,帶你破解...
    沈念sama閱讀 206,311評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件眉睹,死亡現(xiàn)場(chǎng)離奇詭異柑肴,居然都是意外死亡舞终,警方通過(guò)查閱死者的電腦和手機(jī)轻庆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)敛劝,“玉大人余爆,你說(shuō)我怎么就攤上這事】涿耍” “怎么了蛾方?”我有些...
    開(kāi)封第一講書(shū)人閱讀 152,671評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)上陕。 經(jīng)常有香客問(wèn)我桩砰,道長(zhǎng),這世上最難降的妖魔是什么释簿? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,252評(píng)論 1 279
  • 正文 為了忘掉前任亚隅,我火速辦了婚禮,結(jié)果婚禮上辕万,老公的妹妹穿的比我還像新娘枢步。我一直安慰自己沉删,他們只是感情好渐尿,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,253評(píng)論 5 371
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著矾瑰,像睡著了一般砖茸。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上殴穴,一...
    開(kāi)封第一講書(shū)人閱讀 49,031評(píng)論 1 285
  • 那天凉夯,我揣著相機(jī)與錄音,去河邊找鬼采幌。 笑死劲够,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的休傍。 我是一名探鬼主播征绎,決...
    沈念sama閱讀 38,340評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼磨取!你這毒婦竟也來(lái)了人柿?” 一聲冷哼從身側(cè)響起柴墩,我...
    開(kāi)封第一講書(shū)人閱讀 36,973評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎凫岖,沒(méi)想到半個(gè)月后江咳,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,466評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡哥放,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,937評(píng)論 2 323
  • 正文 我和宋清朗相戀三年歼指,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片甥雕。...
    茶點(diǎn)故事閱讀 38,039評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡东臀,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出犀农,到底是詐尸還是另有隱情惰赋,我是刑警寧澤,帶...
    沈念sama閱讀 33,701評(píng)論 4 323
  • 正文 年R本政府宣布呵哨,位于F島的核電站赁濒,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏孟害。R本人自食惡果不足惜拒炎,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,254評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望挨务。 院中可真熱鬧击你,春花似錦、人聲如沸谎柄。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,259評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)朝巫。三九已至鸿摇,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間劈猿,已是汗流浹背拙吉。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,485評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留揪荣,地道東北人筷黔。 一個(gè)月前我還...
    沈念sama閱讀 45,497評(píng)論 2 354
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像仗颈,于是被迫代替她去往敵國(guó)和親佛舱。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,786評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容

  • DOM事件 在我們的日常的生活中,無(wú)時(shí)無(wú)刻不在發(fā)生著各種類型的事件名眉。比如說(shuō)體育賽事粟矿、娛樂(lè)新聞、戰(zhàn)爭(zhēng)甚至天氣變化等等...
    七里之境閱讀 7,381評(píng)論 1 3
  • 在實(shí)現(xiàn)事件委托之前损拢,首先必須先了解事件的綁定陌粹、監(jiān)聽(tīng)和派發(fā)的含義。 事件的綁定 這是最直接簡(jiǎn)單的綁定在DOM元素上的...
    饑人谷_Dylan閱讀 713評(píng)論 0 49
  • 1福压、如何監(jiān)聽(tīng)事件 目前W3C對(duì)DOM進(jìn)行標(biāo)準(zhǔn)化規(guī)定中對(duì)事件監(jiān)聽(tīng)有兩種方式掏秩,DOM level 0 中規(guī)定 butt...
    字母31閱讀 384評(píng)論 0 0
  • 1 什么是事件委托? 用個(gè)例子就可以很簡(jiǎn)單的解釋事件委托是怎么一回事了: 假設(shè)一個(gè)公司有三個(gè)職員在網(wǎng)上買了東西,那...
    流光號(hào)船長(zhǎng)閱讀 562評(píng)論 0 1
  • 我是在劇場(chǎng)看戲時(shí)見(jiàn)到她的。她向我招了招手荆姆,我趁幕間休息的時(shí)候走了過(guò)去蒙幻,在她旁邊坐下。我最后一次見(jiàn)到她還是很久以前的...
    那花閱讀 602評(píng)論 0 3