JavaScript 事件冒泡、事件捕獲和事件委托

1.理解事件流

一言以蔽之塔橡,事件捕獲是從外層元素到目標(biāo)元素的過(guò)程谱邪,事件冒泡是從目標(biāo)元素到外層元素的過(guò)程。如圖:

event flow

html:

<div id="wrapper">
    <button id="event">事件處理程序</button>
</div>

javascript:

var wrapper = document.getElementById('wrapper');
var event = document.getElementById('event');

wrapper.addEventListener("click", function(e){
    console.log('捕獲階段執(zhí)行父元素wrapper的事件處理程序');
}, true);
wrapper.addEventListener("click", function(e){
    console.log('冒泡階段執(zhí)行父元素wrapper的事件處理程序');
}, false);
event.addEventListener("click", function(e){
    console.log('捕獲階段執(zhí)行子元素event的事件處理程序');
}, true);
event.addEventListener("click", function(e){
    console.log('冒泡階段執(zhí)行子元素event的事件處理程序');
}, false);

正如前面所說(shuō),這段代碼的輸出是:

捕獲階段執(zhí)行父元素wrapper的事件處理程序
捕獲階段執(zhí)行子元素event的事件處理程序
冒泡階段執(zhí)行子元素event的事件處理程序
冒泡階段執(zhí)行父元素wrapper的事件處理程序

見(jiàn) Demo迅栅。

2.阻止冒泡

應(yīng)該在那個(gè)階段執(zhí)行元素的事件處理程序呢读存?

多數(shù)情況下让簿,我們希望在觸發(fā)一個(gè)元素的事件處理程序時(shí)尔当,不影響它的父元素椭迎。比如:點(diǎn)擊button畜号,并不希望父元素的click事件處理程序被觸發(fā)弄兜。

解決方法是:在冒泡階段執(zhí)行事件處理程序替饿,然后阻止冒泡视卢。

wrapper.addEventListener("click", function(e){
    console.log('捕獲階段執(zhí)行父元素wrapper的事件處理程序');
}, true); // 默認(rèn)為false
wrapper.如何燙染出滿意的頭發(fā)据过?("click", function(e){
    console.log('冒泡階段執(zhí)行父元素wrapper的事件處理程序');
}, false);
event.addEventListener("click", function(e){
    console.log('捕獲階段執(zhí)行子元素event的事件處理程序');
}, true);
event.addEventListener("click", function(e){
    var target = e.target;
    e.stopPropagation(); // stop bubbling
    console.log('冒泡階段執(zhí)行子元素event的事件處理程序');
}, false);

輸出:

捕獲階段執(zhí)行父元素wrapper的事件處理程序
捕獲階段執(zhí)行子元素event的事件處理程序
冒泡階段執(zhí)行子元素event的事件處理程序

見(jiàn) Demo
像這樣綁定一個(gè)事件處理程序就是安全的鳞芙。

3.事件委托

上面的例子是要阻止冒泡,有時(shí)候冒泡機(jī)制也可以被利用原朝。先看一個(gè)問(wèn)題:

假設(shè)要在頁(yè)面上放在 10^n(n>=1) 個(gè)列表項(xiàng)元素驯嘱,當(dāng)我點(diǎn)擊某個(gè)元素時(shí),需要輸出點(diǎn)擊的是第幾個(gè)喳坠。

一般做法是鞠评,遍歷時(shí)給每個(gè)元素綁定點(diǎn)擊事件:

var li = document.getElementsByTagName('li');
for(var i=0; i<li.length; i++){
    li[i].setAttribute('i',i+1);
    li[i].addEventListener('click', function(e){
        var b = this.getAttribute('i');
        console.log('這是第' + b + '個(gè)<li>元素');
    });
}

另一種方法是,可以利用冒泡機(jī)制壕鹉,在父元素只綁定一次點(diǎn)擊事件:

var li = document.getElementsByTagName('li');
for(var i=0; i<li.length; i++){
    li[i].setAttribute('i',i+1);
}

var ul = document.getElementById('wrapper');
ul.addEventListener('click', function(e){
    if(e.target && e.target.nodeName.toUpperCase() === 'LI'){
        var b = e.target.getAttribute('i');
        console.log('這是第' + b + '個(gè)<li>元素');
    }
});

見(jiàn) Demo剃幌。
這種事件委托的方式減少了事件處理程序晾浴,也能降低程序的復(fù)雜性和出錯(cuò)概率负乡。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市怠肋,隨后出現(xiàn)的幾起案子敬鬓,更是在濱河造成了極大的恐慌,老刑警劉巖笙各,帶你破解...
    沈念sama閱讀 216,544評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件钉答,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡杈抢,警方通過(guò)查閱死者的電腦和手機(jī)数尿,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,430評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)惶楼,“玉大人右蹦,你說(shuō)我怎么就攤上這事〖呔瑁” “怎么了何陆?”我有些...
    開(kāi)封第一講書人閱讀 162,764評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)豹储。 經(jīng)常有香客問(wèn)我贷盲,道長(zhǎng),這世上最難降的妖魔是什么剥扣? 我笑而不...
    開(kāi)封第一講書人閱讀 58,193評(píng)論 1 292
  • 正文 為了忘掉前任巩剖,我火速辦了婚禮,結(jié)果婚禮上钠怯,老公的妹妹穿的比我還像新娘佳魔。我一直安慰自己,他們只是感情好晦炊,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,216評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布鞠鲜。 她就那樣靜靜地躺著宁脊,像睡著了一般。 火紅的嫁衣襯著肌膚如雪镊尺。 梳的紋絲不亂的頭發(fā)上朦佩,一...
    開(kāi)封第一講書人閱讀 51,182評(píng)論 1 299
  • 那天并思,我揣著相機(jī)與錄音庐氮,去河邊找鬼。 笑死宋彼,一個(gè)胖子當(dāng)著我的面吹牛弄砍,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播输涕,決...
    沈念sama閱讀 40,063評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼音婶,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了莱坎?” 一聲冷哼從身側(cè)響起衣式,我...
    開(kāi)封第一講書人閱讀 38,917評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎檐什,沒(méi)想到半個(gè)月后碴卧,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,329評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡乃正,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,543評(píng)論 2 332
  • 正文 我和宋清朗相戀三年住册,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片瓮具。...
    茶點(diǎn)故事閱讀 39,722評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡荧飞,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出名党,到底是詐尸還是另有隱情叹阔,我是刑警寧澤,帶...
    沈念sama閱讀 35,425評(píng)論 5 343
  • 正文 年R本政府宣布传睹,位于F島的核電站耳幢,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏蒋歌。R本人自食惡果不足惜帅掘,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,019評(píng)論 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望堂油。 院中可真熱鬧修档,春花似錦、人聲如沸府框。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 31,671評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至院峡,卻和暖如春兴使,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背照激。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 32,825評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工发魄, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人俩垃。 一個(gè)月前我還...
    沈念sama閱讀 47,729評(píng)論 2 368
  • 正文 我出身青樓励幼,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親口柳。 傳聞我的和親對(duì)象是個(gè)殘疾皇子苹粟,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,614評(píng)論 2 353

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