4月12日學習筆記——js事件冒泡機制

来源:http://www.cnblogs.com/little-jelly/archive/2016/04/12/5383573.html
-Advertisement-
Play Games

頁面如下: 我們現在想實現這樣的功能,在div1 點擊的時候,彈出 "你好,我是最外層div。",點擊div2 的時候,彈出 "你好,我是第二層div";點擊span 的時候,彈出"您好,我是span。"。 但是當點擊span時,三個對話框會先後彈出來,無論是body,body 的子元素div1,還 ...


 1 <!doctype html>
 2 <html lang="zh">
 3   <head>
 4     <meta charset="utf-8"/>
 5     <title>Hello World jQuery!</title>
 6     <script src="scripts/jquery-1.5.1.js"></script>
 7     <script type="text/javascript">
 8     window.onload = function() {  
 9         document.getElementById("box1").onclick=function(event){  
10             alert("您好,我是最外層div。");  
11         }  
12         document.getElementById("box2").onclick=function(event){  
13             alert("您好,我是第二層div。");  
14         }
15         document.getElementById("span").onclick=function(event){  
16             alert("您好,我是span。");  
17         }
18     }  
19     </script>
20     <style type="text/css">  
21 .box1 {  
22     border: green 40px solid;  
23     width: 300px;  
24     height: 300px;  
25     margin: auto;  
26 }  
27   
28 .box2 {  
29     border: yellow 40px solid;  
30     width: 220px;  
31     height: 220px;  
32     margin: auto;  
33 }  
34   
35 span {  
36     position: relative;  
37     left: 50px;  
38     top: 50px;  
39     background-color: rgba(128, 128, 128, 0.22);  
40 }  
41 </style> 
42   </head>
43   <body id="body">
44     <div id="box1" class="box1">  
45         <div id="box2" class="box2">  
46             <span id="span">This is a span.</span>  
47         </div>  
48     </div>
49   </body>
50 </html>

頁面如下:

 我們現在想實現這樣的功能,在div1 點擊的時候,彈出 "你好,我是最外層div。",點擊div2 的時候,彈出 "你好,我是第二層div";點擊span 的時候,彈出"您好,我是span。"。

但是當點擊span時,三個對話框會先後彈出來,無論是body,body 的子元素div1,還是 div1的子元素div2,還有 span, 當這些元素被點擊click時,都會產生click事件,並且body都會捕獲到,然後調用相應的事件處理函數。就像水中的氣泡從底往上冒一樣,事件也會往上傳遞。所以原因就在於事件的冒泡,點擊span的時候,span 會把產生的事件往上冒泡,作為父節點的div2 和 祖父節點的div1也會收到此事件,於是會做出事件響應,執行響應函數。

解決辦法:

辦法一:在相應的處理函數內,加入  event.stopPropagation()   ,終止事件的廣播分發,這樣事件停留在本節點,不會再往外傳播了。修改上述的script片段:

 1 <script type="text/javascript">  
 2     window.onload = function() {  
 3         document.getElementById("box1").addEventListener("click",function(event){  
 4             alert("您好,我是最外層div。");  
 5             event.stopPropagation();  
 6         });  
 7         document.getElementById("box2").addEventListener("click",function(event){  
 8             alert("您好,我是第二層div。");  
 9             event.stopPropagation();  
10         });  
11         document.getElementById("span").addEventListener("click",function(event){  
12             alert("您好,我是span。");  
13             event.stopPropagation();  
14         });  
15     }  
16 </script>

 

辦法二:event.target 引用了產生此event對象的dom 節點,而event.currrentTarget 則引用了當前處理節點,我們可以通過這 兩個target 是否相等。

            比如span 點擊事件,產生一個event 事件對象,event.target 指向了span元素,span處理此事件時,event.currentTarget 指向的也是span元素,這時判斷兩者相等,則執行相應的處理函數。而事件傳遞給 div2 的時候,event.currentTarget變成 div2,這時候判斷二者不相等,即事件不是div2 本身產生的,就不作響應處理邏輯。

 1 <script type="text/javascript">  
 2     window.onload = function() {  
 3         document.getElementById("box1").addEventListener("click",function(event){  
 4             if(event.target == event.currentTarget)  
 5             {  
 6                 alert("您好,我是最外層div。");  
 7             }  
 8         });  
 9         document.getElementById("box2").addEventListener("click",function(event){  
10             if(event.target == event.currentTarget)  
11             {  
12                 alert("您好,我是第二層div。");  
13             }  
14         });  
15         document.getElementById("span").addEventListener("click",function(event){  
16             if(event.target == event.currentTarget)  
17             {  
18                 alert("您好,我是span。");  
19                   
20             }  
21         });  
22     }  
23 </script>

 

兩種方法的缺點:

      方法一缺點:為了實現點擊特定的元素顯示對應的信息,方法一要求每個元素的子元素也必須終止事件的冒泡傳遞,即跟別的元素功能上強關聯,這樣的方法會很脆弱。比如,如果span 元素的處理函數沒有執行冒泡終止,則事件會傳到div2 上,這樣會造成div2 的提示信息;

      方法二缺點:方法二為每一個元素都增加了事件監聽處理函數,事件的處理邏輯都很相似,即都有判斷 if(event.target == event.currentTarget),這樣存在了很大的代碼冗餘,現在是三個元素還好,當有10幾個,上百個又該怎麼辦呢?還有就是為每一個元素都有處理函數,在一定程度上增加邏輯和代碼的複雜度。

改進方法二:

既然事件是冒泡傳遞的,那可以讓某個父節點統一處理事件,通過判斷事件的發生地(即事件產生的節點),然後做出相應的處理。

 1 <script type="text/javascript">  
 2     window.onload = function() {  
 3         document.getElementById("body").addEventListener("click",eventPerformed);  
 4     }  
 5     function eventPerformed(event) {  
 6         var target = event.target;  
 7         switch (target.id) {  
 8         case "span":   
 9             alert("您好,我是span。");  
10             break;  
11         case "div1":  
12             alert("您好,我是第二層div。");  
13             break;  
14         case "div2":  
15              alert("您好,我是最外層div。");  
16             break;  
17         }  
18     }  
19 </script>  

通過以上方式,我們把本來每個元素都要有的處理函數,都交給了其祖父節點body 元素來完成了,也就是說,span,div2,div1 將自己的響應邏輯委托給body,讓它來完成相應邏輯,自己不實現相應邏輯,這個模式,就是所謂的事件委托


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 裝飾模式(Decorator) 定義 裝飾模式(Decorator),動態地給一個對象添加一些額外的職責,就增加功能來說,裝飾模式比生成子類更為靈活。 類圖 描述 Component:被裝飾者和裝飾者共有的基類; ConcreteComponent:被裝飾者的具體類; Decorator:裝飾類,包 ...
  • 編者按 與傳統意義上的紅包相比,近兩年火起來的“紅包”,似乎才是如今春節的一大重頭戲。歷經上千年時代傳承與變遷,春節發紅包早已成為歷史沉澱的文化習俗,融入了民族的血脈。按照各家公佈的數據,除夕全天微信用戶紅包總發送量達到80.8億個,紅包峰值收發量為40.9萬個/秒。春晚直播期間討論春晚的微博達到5 ...
  • 在配置cat集群時,有一些設置是我們應該註意的,從它的部署文檔中我們可以看到相關信息,但說的還不夠明確和重要,大叔今天總結一下Cat集群配置的註意事項 1 服務端datasources.xml用來設置連接的mysql,集群里的伺服器對這項配置是相同的 2 服務端的server.xml,用來設置集群里 ...
  • 在之前發佈Objective-C系列博客的時候,其中提到過OC的通知機制,請參考《Objective-C中的老闆是這樣發通知的(Notification)》這篇博客。在之前關於Notification的博客中,只介紹了Foundation框架中的通知的使用方式。正如前面博客中提到的那樣,通知是“一對 ...
  • 上一篇寫了nginx負載均衡,此篇實現高可用(HA)。系統整體設計是採用Nginx做負載均衡,若出現Nginx單機故障,則導致整個系統無法正常運行。針對系統架構設計的高可用要求,我們需要解決Nginx負載均衡出現單機故障時,系統正常運行的需求。所以系統架構引入Keepalived組件,實現系統高可用 ...
  • 組合模式(Composite) 定義 組合模式(Composite),將對象組合成樹形結構以表示“部分-整體”的層次結構。組合模式使得用戶對單個對象和組合對象的使用具有一致性。 組合模式有兩種形式:透明方式的組合模式和安全方式的組合模式。 類圖 描述 Component:它可以是介面或抽象類,為葉子 ...
  • 乾貨|建議初創團隊起初也要構建分散式應用 本文內容整理自W-Time技術分享沙龍-天津站現場演講《一切都是分佈的》,演講者:李傲,問啊聯合創始人,前中交車聯網總架構。 好多人都會問什麼是架構師?其實架構師的定義很寬泛,前端後端的定義都不一樣。作為後端出身的架構師,我認為後端並不是大家想的封裝組件,它 ...
  • 網頁可見區域寬: document.body.clientWidth網頁可見區域高: document.body.clientHeight網頁可見區域寬: document.body.offsetWidth (包括邊線的寬)網頁可見區域高: document.body.offsetHeight (包 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...