自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

淺析JavaScript的事件代理和委托

開發(fā) 前端
在javasript中delegate這個詞經(jīng)常出現(xiàn),看字面的意思,代理、委托。那么它究竟在什么樣的情況下使用?它的原理又是什么?在各種框架中,也經(jīng)常能看到delegate相關(guān)的接口。這些接口又有什么特殊的用法呢?這篇文章就主要介紹一下javascript delegate的用法和原理,以及Dojo,jQuery等框架中delegate的接口。

在javasript中delegate這個詞經(jīng)常出現(xiàn),看字面的意思,代理、委托。那么它究竟在什么樣的情況下使用?它的原理又是什么?在各種框架中,也經(jīng)常能看到delegate相關(guān)的接口。這些接口又有什么特殊的用法呢?這篇文章就主要介紹一下javascript delegate的用法和原理,以及Dojo,jQuery等框架中delegate的接口。

JavaScript事件代理

首先介紹一下JavaScript的事件代理。事件代理在JS世界中一個非常有用也很有趣的功能。當我們需要對很多元素添加事件的時候,可以通過將事件添加到它們的父節(jié)點而將事件委托給父節(jié)點來觸發(fā)處理函數(shù)。這主要得益于瀏覽器的事件冒泡機制,后面會詳細介紹。下面我們具體舉個例子來解釋如何使用這個特性。這個例子主要取自David Walsh的相關(guān)文章(How JavaScript Event Delegation Works)。

假設(shè)有一個 UL 的父節(jié)點,包含了很多個 Li 的子節(jié)點:

  1. <ul id="parent-list"
  2.   <li id="post-1">Item 1</li> 
  3.   <li id="post-2">Item 2</li> 
  4.   <li id="post-3">Item 3</li> 
  5.   <li id="post-4">Item 4</li> 
  6.   <li id="post-5">Item 5</li> 
  7.   <li id="post-6">Item 6</li> 
  8. </ul>  

當我們的鼠標移到Li上的時候,需要獲取此Li的相關(guān)信息并飄出懸浮窗以顯示詳細信息,或者當某個Li被點擊的時候需要觸發(fā)相應(yīng)的處理事件。我們通常的寫法,是為每個Li都添加一些類似onMouseOver或者onClick之類的事件監(jiān)聽。

  1. function addListeners4Li(liNode){ 
  2.     liNode.onclick = function clickHandler(){...}; 
  3.     liNode.onmouseover = function mouseOverHandler(){...} 
  4.  
  5. window.onload = function(){ 
  6.     var ulNode = document.getElementById("parent-list"); 
  7.     var liNodes = ulNode.getElementByTagName("Li"); 
  8.     for(var i=0, l = liNodes.length; i < l; i++){ 
  9.         addListeners4Li(liNodes[i]); 
  10.     }    
  11.  

如果這個UL中的Li子元素會頻繁地添加或者刪除,我們就需要在每次添加Li的時候都調(diào)用這個addListeners4Li方法來為每個Li節(jié)點添加事件處理函數(shù)。這就添加的復雜度和出錯的可能性。

更簡單的方法是使用事件代理機制,當事件被拋到更上層的父節(jié)點的時候,我們通過檢查事件的目標對象(target)來判斷并獲取事件源Li。下面的代碼可以完成我們想要的效果:

  1. // 獲取父節(jié)點,并為它添加一個click事件 
  2. document.getElementById("parent-list").addEventListener("click",function(e) { 
  3.   // 檢查事件源e.targe是否為Li 
  4.   if(e.target && e.target.nodeName.toUpperCase == "LI") { 
  5.     // 真正的處理過程在這里 
  6.     console.log("List item ",e.target.id.replace("post-")," was clicked!"); 
  7.   } 
  8. });  

為父節(jié)點添加一個click事件,當子節(jié)點被點擊的時候,click事件會從子節(jié)點開始向上冒泡。父節(jié)點捕獲到事件之后,通過判斷e.target.nodeName來判斷是否為我們需要處理的節(jié)點。并且通過e.target拿到了被點擊的Li節(jié)點。從而可以獲取到相應(yīng)的信息,并作處理。

事件冒泡及捕獲

之前的介紹中已經(jīng)說到了瀏覽器的事件冒泡機制。這里再詳細介紹一下瀏覽器處理DOM事件的過程。對于事件的捕獲和處理,不同的瀏覽器廠商有不同的處理機制,這里我們主要介紹W3C對DOM2.0定義的標準事件。

DOM2.0模型將事件處理流程分為三個階段:一、事件捕獲階段,二、事件目標階段,三、事件起泡階段。如圖:

 事件捕獲:當某個元素觸發(fā)某個事件(如onclick),頂層對象document就會發(fā)出一個事件流,隨著DOM樹的節(jié)點向目標元素節(jié)點流去,直到到達事件真正發(fā)生的目標元素。在這個過程中,事件相應(yīng)的監(jiān)聽函數(shù)是不會被觸發(fā)的。

事件目標:當?shù)竭_目標元素之后,執(zhí)行目標元素該事件相應(yīng)的處理函數(shù)。如果沒有綁定監(jiān)聽函數(shù),那就不執(zhí)行。

事件起泡:從目標元素開始,往頂層元素傳播。途中如果有節(jié)點綁定了相應(yīng)的事件處理函數(shù),這些函數(shù)都會被一次觸發(fā)。如果想阻止事件起泡,可以使用e.stopPropagation()(Firefox)或者e.cancelBubble=true(IE)來組織事件的冒泡傳播。

jQuery和Dojo中delegate函數(shù)

下面看一下Dojo和jQuery中提供的事件代理接口的使用方法。

  1. $("#link-list").delegate("a""click"function(){ 
  2.   // "$(this)" is the node that was clicked 
  3.   console.log("you clicked a link!",$(this)); 
  4. });  

jQuery的delegate的方法需要三個參數(shù),一個選擇器,一個時間名稱,和事件處理函數(shù)。

而Dojo的與jQuery相似,僅是兩者的編程風格上的差別:

  1. require(["dojo/query","dojox/NodeList/delegate"], function(query,delegate){ 
  2.  
  3.     query("#link-list").delegate("a","onclick",function(event) { 
  4.     // "this.node" is the node that was clicked 
  5.     console.log("you clicked a link!",this); 
  6.   }); 
  7. })  

Dojo的delegate模塊在dojox.NodeList中,提供的接口與jQuery一樣,參數(shù)也相同。

優(yōu)點通過上面的介紹,大家應(yīng)該能夠體會到使用事件委托對于web應(yīng)用程序帶來的幾個優(yōu)點:

1.管理的函數(shù)變少了。不需要為每個元素都添加監(jiān)聽函數(shù)。對于同一個父節(jié)點下面類似的子元素,可以通過委托給父元素的監(jiān)聽函數(shù)來處理事件。

2.可以方便地動態(tài)添加和修改元素,不需要因為元素的改動而修改事件綁定。

3.JavaScript和DOM節(jié)點之間的關(guān)聯(lián)變少了,這樣也就減少了因循環(huán)引用而帶來的內(nèi)存泄漏發(fā)生的概率。

寫到這里,突然想起了之前對于Dojo DataGrid的困惑:那么多的rows和cells,如何處理他們事件之間的關(guān)系。現(xiàn)在想想,使用委托就很簡單了。所有的事件委托到grid最外層的節(jié)點上,當事件發(fā)生的時候通過一些方法來獲取和添加事件的額外屬性,如rowIndex, cellIndex,之后在分配到onRowClick,onCellClick之類的處理函數(shù)上。

在JavaScript編程中使用代理

上面介紹的是對DOM事件處理時,利用瀏覽器冒泡機制為DOM元素添加事件代理。其實在純JS編程中,我們也可以使用這樣的編程模式,來創(chuàng)建代理對象來操作目標對象。這里引用司徒正美相關(guān)文章中的一個例子:

  1. var delegate = function(client, clientMethod) { 
  2.         return function() { 
  3.             return clientMethod.apply(client, arguments); 
  4.         } 
  5.     } 
  6.     var ClassA = function() { 
  7.         var _color = "red"
  8.         return { 
  9.             getColor: function() { 
  10.                 console.log("Color: " + _color); 
  11.             }, 
  12.             setColor: function(color) { 
  13.                 _color = color; 
  14.             } 
  15.         }; 
  16.     }; 
  17.  
  18.     var a = new ClassA(); 
  19.     a.getColor(); 
  20.     a.setColor("green"); 
  21.     a.getColor(); 
  22.     console.log("執(zhí)行代理!"); 
  23.     var d = delegate(a, a.setColor); 
  24.     d("blue"); 
  25.     console.log("執(zhí)行完畢!"); 
  26.     a.getColor();  

上面的例子中,通過調(diào)用delegate()函數(shù)創(chuàng)建的代理函數(shù)d來操作對a的修改。這種方式盡管是使用了apply(call也可以)來實現(xiàn)了調(diào)用對象的轉(zhuǎn)移,但是從編程模式上實現(xiàn)了對某些對象的隱藏,可以保護這些對象不被隨便訪問和修改。

在很多框架中都引用了委托這個概念用來指定方法的運行作用域。比較典型的如dojo.hitch(scope,method)和ExtJS的createDelegate(obj,args)。有興趣的同學可以看一下他們的源代碼,主要也是js函數(shù)的apply方法來制定執(zhí)行作用域。

責任編輯:龐桂玉 來源: segmentfault
相關(guān)推薦

2009-10-09 09:07:40

C#委托和事件

2011-06-16 14:38:18

JavaScript事件委托

2021-07-27 22:56:00

JavaScript編程開發(fā)

2009-01-19 09:40:53

JavaScript事件代理事件處理器

2011-06-16 15:14:17

VB.NET事件委托

2021-03-16 10:00:40

JavaScript用戶登錄表單JavaScript基

2021-10-19 09:31:19

Javascript 登錄表單前端

2024-12-26 16:13:53

JavaScript開發(fā)表單

2009-08-18 10:54:17

C#事件和委托

2009-08-20 18:37:52

委托C#異步委托

2009-08-27 16:53:01

C#委托C#事件

2024-06-28 10:19:02

委托事件C#

2009-08-31 18:34:57

C#接口事件

2009-08-26 14:27:54

C#委托和事件

2009-08-18 11:08:24

.Net Framew

2022-07-28 08:34:59

事件委托JS

2024-05-16 13:36:04

C#委托事件

2009-08-31 09:20:37

C#事件注冊和注銷

2009-09-10 15:32:54

事件與委托

2009-08-03 13:23:04

C#編程組件-事件-委托
點贊
收藏

51CTO技術(shù)棧公眾號