理解DOM事件流的三個階段 - Lxxyx的開發筆記 - SegmentFault 思否

本文主要解決兩個問題:

  1. 什么是事件流

  2. DOM事件流的三個階段

起因

在學習前端的大半年來,對DOM事件了解甚少。一般也只是用用onclick來綁定個點擊事件。在寒假深入學習JavaScript時,愈發覺得自己對DOM事件了解不夠,遂打開我的《JavaScript高級程序設計》,翻到DOM事件那一章,開始第二次學習之旅。
當然,DOM事件所囊括的知識較為龐雜,所以本文專注與自己學習時所碰到的難點,DOM事件流。

流的概念,在現今的JavaScript中隨處可見。比如說React中的單向數據流,Node中的流,又或是今天本文所講的DOM事件流。都是流的一種生動體現。
至于流的具體概念,我們采用下文的解釋:

用術語說流是對輸入輸出設備的抽象。以程序的角度說,流是具有方向的數據。
通通連起來——無處不在的流 淘寶FED--愈之

事件流之事件冒泡與事件捕獲

在瀏覽器發展的過程中,開發團隊遇到了一個問題。那就是頁面中的哪一部分擁有特定的事件?
可以想象畫在一張紙上的一組同心圓,如果你把手指放在圓心上,那么你的手指指向的其實不是一個圓,而是紙上所有的圓。放到實際頁面中就是,你點擊一個按鈕,事實上你還同時點擊了按鈕所有的父元素。
開發團隊的問題就在于,當點擊按鈕時,是按鈕最外層的父元素先收到事件并執行,還是具體元素先收到事件并執行?所以這兒引入了事件流的概念。

事件流所描述的就是從頁面中接受事件的順序。

因為有兩種觀點,所以事件流也有兩種,分別是事件冒泡和事件捕獲。現行的主流是事件冒泡。

事件冒泡

事件冒泡即事件開始時,由最具體的元素接收(也就是事件發生所在的節點),然后逐級傳播到較為不具體的節點。
舉個栗子,就很容易明白了。

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Event Bubbling</title>
</head>
<body><button id="clickMe">Click Me</button>
</body>
</html>

然后,我們給button和它的父元素,加入點擊事件。

var button = document.getElementById('clickMe');button.onclick = function() {console.log('1. You click Button');
};
document.body.onclick = function() {console.log('2. You click body');
};
document.onclick = function() {console.log('3. You click document');
};
window.onclick = function() {console.log('4. You click window');
};

效果如圖所示:

在代碼所示的頁面中,如果點擊了button,那么這個點擊事件會按如下的順序傳播(Chrome瀏覽器):

  1. button

  2. body

  3. document

  4. window

也就是說,click事件首先在<button>元素上發生,然后逐級向上傳播。這就是事件冒泡。

事件捕獲

事件捕獲的概念,與事件冒泡正好相反。它認為當某個事件發生時,父元素應該更早接收到事件,具體元素則最后接收到事件。比如說剛才的demo,如果是事件捕獲的話,事件發生順序會是這樣的:

  1. window

  2. document

  3. body

  4. button


當然,由于時代更迭,事件冒泡方式更勝一籌。所以放心的使用事件冒泡,有特殊需要再使用事件捕獲即可。

DOM事件流

DOM事件流包括三個階段。

  1. 事件捕獲階段

  2. 處于目標階段

  3. 事件冒泡階段

如圖所示(圖片源于網絡,若侵權請告知):

1. 事件捕獲階段

也就是說,當事件發生時,首先發生的是事件捕獲,為父元素截獲事件提供了機會。
例如,我把上面的Demo中,window點擊事件更改為使用事件捕獲模式。(addEventListener最后一個參數,為true則代表使用事件捕獲模式,false則表示使用事件冒泡模式。不理解的可以去學習一下addEventListener函數的使用)

window.addEventListener('click', function() {console.log('4. You click window');
}, true);

此時,點擊button的效果是這樣的。
?

可以看到,點擊事件先被父元素截獲了,且該函數只在事件捕獲階段起作用。

處于目標與事件冒泡階段

事件到了具體元素時,在具體元素上發生,并且被看成冒泡階段的一部分。
隨后,冒泡階段發生,事件開始冒泡。

阻止事件冒泡

事件冒泡過程,是可以被阻止的。防止事件冒泡而帶來不必要的錯誤和困擾。
這個方法就是:stopPropagation()
我們對button的click事件做一些改造。

button.addEventListener('click', function(event) {// event為事件對象console.log('1. You click Button');event.stopPropagation();console.log('Stop Propagation!');
}, false);

點擊后,效果如下圖:

不難看出,事件在到達具體元素后,停止了冒泡。但不影響父元素的事件捕獲。

總結與感想

事件流:描述的就是從頁面中接受事件的順序。分有事件冒泡與事件捕獲兩種。
DOM事件流的三個階段:

  1. 事件捕獲階段

  2. 處于目標階段

  3. 事件冒泡階段

在學習DOM事件的過程中,了解了DOM事件的三個階段,也知道事件冒泡是干啥用的,又如何阻止。配合前期所學的二叉樹的相關知識,受益匪淺。

前端路漫漫,且行且歌~
最后附上本人博客地址和原文鏈接,希望能與各位多多交流。

?

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/248984.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/248984.shtml
英文地址,請注明出處:http://en.pswp.cn/news/248984.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

支付寶APP支付 統一下單 php服務端 tp5

/*支付寶第三方支付 * *生成APP支付訂單信息 * param number  uid 用戶id * param string   token 用戶token * param number  oid 訂單id * param string   title 標題 * param string    body …

python第十九天(random、json、pickle、hashlib、hmac、shutil、shevle模塊)

今日內容&#xff1a; 1. random 模塊 2. json模塊 3. pickle 模塊 4.hashlib 模塊 5. hmac 模塊 6. shutil 模塊 7. shelve 模塊 1. random 模塊&#xff1a; random 模塊 獲取隨機值import randomfor i in range(10): print(random.random()) # random.random() 隨機獲取…

NodeJS入門04-Express路由和中間件 - 小之 - 博客園

nodeJS入門04-Express路由和中間件 Express框架是后臺的Node框架&#xff0c;在后臺的受歡迎的程度&#xff0c;和jQuery一樣&#xff0c;就是企業的事實上的標準。 路由 路由是指如何定義應用的端點&#xff08;URIs&#xff09;以及如何響應客戶端的請求。 路由是由一個 …

jmeter(四十五)常用Beanshell腳本

整理了一批jmeter常用的beanshell腳本供大家參考&#xff01; 時間戳 import java.text.SimpleDateFormat; import java.util.Calendar; import java.util.Date; try{ Date date new Date(); //獲取當前時間 SimpleDateFormat sf new SimpleDateFormat("yyyy-MM-dd HH:mm…

Critical error detected c0000374

我發現出現上述錯誤是 free 兩次內存 float* ddnew float[2];delete[] dd;delete[] dd;轉載于:https://www.cnblogs.com/hook-gou/p/9994662.html

nodejs開發 過程中express路由與中間件的理解 - pyj063 - 博客園

nodejs開發 過程中express路由與中間件的理解 nodejs開發 express路由與中間件 路由 通常HTTP URL的格式是這樣的&#xff1a; http://host[:port][path] http表示協議。 host表示主機。 port為端口&#xff0c;可選字段&#xff0c;不提供時默認為80。 path指定請求資源的…

錯誤MSB4018 “ResolvePackageAssets”任務意外失敗的解決方法

昨天系統奔潰了&#xff0c;重裝系統后發現&#xff0c;之前寫的.netcore項目打開后重新生成報錯&#xff0c;錯誤如下嚴重性 代碼 說明 項目 文件 行 禁止顯示狀態 錯誤 MSB4018 “ResolvePackageAssets”任務意外失敗。 NuGet.Packaging.Core.Packag…

(五)Unity插件生成

1&#xff09;新建空的AndroidStudio工程&#xff0c;但是新建過程時最小SDK版本要與unity一致&#xff0c;如下圖所示&#xff0c;本次操作均為api16 2&#xff09;創建Library&#xff0c;如下圖所示&#xff0c;新建module&#xff0c;然后選擇Android Library。 新建模塊為…

centeros7安裝mysql - 風中追風_lonely - 博客園

轉載自&#xff1a;https://www.linuxidc.com/Linux/2016-09/135288.htm 安裝之前先安裝基本環境&#xff1a;yum install -y perl perl-Module-Build net-tools autoconf libaio numactl-libs 1、配置YUM源 在MySQL官網中下載YUM源rpm安裝包&#xff1a;http://dev.mysql.c…

失物招領平臺6

昨天做了什么&#xff1a;學習了復選框、列表視圖、網絡視圖&#xff0c;集體討論了登錄頁面的構思 今天準備做什么&#xff1a;繼續學習Android stdio&#xff0c;改善登錄頁面 遇到的問題&#xff1a;時間遠遠不夠。轉載于:https://www.cnblogs.com/sljslj/p/11056074.html

前后端交互json字符串

//將需要的參數轉成json字符串&#xff0c;然后用utf-8編碼 var obj encodeURIComponent(JSON.stringify(this.categories),"utf-8") //后臺將前臺的json字符串按照utf-8的格式解碼&#xff0c;然后進行轉換 RequestMapping(value "/updateMaterialDemoInfo.d…

CSS布局解決方案(終結版)

前端布局非常重要的一環就是頁面框架的搭建&#xff0c;也是最基礎的一環。在頁面框架的搭建之中&#xff0c;又有居中布局、多列布局以及全局布局&#xff0c;今天我們就來總結總結前端干貨中的CSS布局。 居中布局 水平居中 1&#xff09;使用inline-blocktext-align &…

個人作業7 第一階段SCRUM沖刺(七)

了解了一下云服務器&#xff0c;下載了阿里云。 然而搞了半天還是沒應用上這個云服務器..轉載于:https://www.cnblogs.com/jbwen/p/11071733.html

Dcloud HTML5 監聽藍牙設備 調用 原生安卓實現 - aspirant - 博客園

最近一直搞Dcloud &#xff0c;這是HTML5版本的開發&#xff0c;打包時候&#xff0c;可以打包成 apk 和ipa 分別運行在安卓和ios 機器上面&#xff0c; 但是這里面的資料很少&#xff0c;遇到問題&#xff0c;之后只能自己鉆研總結&#xff0c; 現在有這么一個需求&#xff…

NOIP2018游記

NOIP 2018 游記 又是一年 \(NOIP\) 呢...第二次參加了,希望這一次能不再擦線吧...畢竟我真的很想去 \(WC\) ,也很想去省選. 最后悔的事就是在初三了,恰逢直升,大好年華,停課學 \(OI\) ,但我竟然在某兩位 \(dalao\) 帶領下搓了一年 爐石 \(\& \: SC2\) &#xff1f;&#xf…

四葉草社交平臺——十天沖刺(5)

今天沒能安排好各種的任務&#xff0c;姑且拍了張照片就散了。 我的任務就是把登錄功能完成&#xff0c;先讓其他人把資源載入問題解決了&#xff0c;然后我再看看動態如何發送。 轉載于:https://www.cnblogs.com/limitCM/p/10925161.html

Django forms組件

校驗字段 模板文件 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title> </head> <body> <form action"" method"post"><p>用戶名: <…

為什么要用TypeScript - 肉豬 - 博客園

為什么要用TypeScript 以下是本人的一點拙見&#xff0c;歡迎指正。 TypeScript的設計目的應該是解決JavaScript的“痛點”&#xff1a;弱類型和沒有命名空間&#xff0c;導致很難模塊化&#xff0c;不適合開發大型程序。另外它還提供了一些語法糖來幫助大家更方便地實踐面向…

java繼承上機作業

實現如下類之間的繼承關系&#xff0c;并編寫Music類來測試這些類。 1 package su;2 3 class Instrument{4 public void play() {5 System.out.println("彈奏樂器");6 }7 8 public void play2() {9 // TODO 自動生成的方法存根 10 …

自定義標簽的作用

1&#xff09;控制標簽體內容是否輸出 2&#xff09;控制標簽余下內容是否輸出 3&#xff09;控制重復輸出標簽體內容 4&#xff09;改變標簽體內容 5&#xff09;帶屬性的標簽 package com.loaderman.demo.a_tag;import java.io.IOException; import java.io.StringWriter;imp…