亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

JavaScript事件代理和委托詳解

 更新時(shí)間:2016年04月08日 08:40:33   作者:花溪立  
這篇文章主要為大家詳細(xì)介紹了JavaScript事件代理和委托,感興趣的小伙伴們可以參考一下

在javasript中,代理、委托經(jīng)常出現(xiàn)。

那么它究竟在什么樣的情況下使用?它的原理又是什么?

這里介紹一下javascript delegate的用法和原理,以及Dojo,jQuery等框架中delegate的接口。

JavaScript事件代理
事件代理在JS世界中一個(gè)非常有用也很有趣的功能。當(dāng)我們需要對(duì)很多元素添加事件的時(shí)候,可以通過將事件添加到它們的父節(jié)點(diǎn)而將事件委托給父節(jié)點(diǎn)來觸發(fā)處理函數(shù)。

這主要得益于瀏覽器的事件冒泡機(jī)制,下面我們具體舉個(gè)例子來解釋如何使用這個(gè)特性。

這個(gè)例子主要取自David Walsh的相關(guān)文章(How JavaScript Event Delegation Works)。

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

<ul id="list">
 <li id="li-1">Li 1</li>
 <li id="li-2">Li 2</li>
 <li id="li-3">Li 3</li>
 <li id="li-4">Li 4</li>
 <li id="li-5">Li 5</li> 
</ul>

當(dāng)我們的鼠標(biāo)移到Li上的時(shí)候,需要獲取此Li的相關(guān)信息并飄出懸浮窗以顯示詳細(xì)信息,或者當(dāng)某個(gè)Li被點(diǎn)擊的時(shí)候需要觸發(fā)相應(yīng)的處理事件。

我們通常的寫法,是為每個(gè)Li都添加一些類似onMouseOver或者onClick之類的事件監(jiān)聽。

function addListenersLi(liElement) {
  liElement.onclick = function clickHandler() {
   //TODO
  };
  liElement.onmouseover = function mouseOverHandler() {
   //TODO
  }
 }

 window.onload = function() {
  var ulElement = document.getElementById("list");
  var liElements = ulElement.getElementByTagName("Li");
   for (var i = liElements.length - 1; i >= 0; i--) {
    addListenersLi(liElements[i]);
   } 
 }

如果這個(gè)UL中的Li子元素會(huì)頻繁地添加或者刪除,我們就需要在每次添加Li的時(shí)候都調(diào)用這個(gè)addListenersLi方法來為每個(gè)Li節(jié)點(diǎn)添加事件處理函數(shù)。

這會(huì)造成添加或者刪除過程的復(fù)雜度和出錯(cuò)的可能性。

解決問題方法是使用事件代理機(jī)制,當(dāng)事件被拋到更上層的父節(jié)點(diǎn)的時(shí)候,我們通過檢查事件的目標(biāo)對(duì)象(target)來判斷并獲取事件源Li。

下面的代碼可以完成想要的效果: 

/ 獲取父節(jié)點(diǎn),并為它添加一個(gè)click事件
document.getElementById("list").addEventListener("click",function(e) {
 // 檢查事件源e.targe是否為L(zhǎng)i
 if(e.target && e.target.nodeName.toUpperCase == "LI") {
 // 
 //TODO
 console.log("List item ",e.target.id," was clicked!");
 }
});

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

事件冒泡及捕獲
瀏覽器的事件冒泡機(jī)制,對(duì)于事件的捕獲和處理,不同的瀏覽器廠商有不同的處理機(jī)制,這里介紹W3C對(duì)DOM2.0定義的標(biāo)準(zhǔn)事件。

DOM2.0模型將事件處理流程分為三個(gè)階段:

一、事件捕獲階段,

二、事件目標(biāo)階段,

三、事件起泡階段。

如下圖:

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

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

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

jQuery和Dojo中delegate函數(shù)
下面看一下Dojo和jQuery中提供的事件代理接口的使用方法。

jQuery:

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

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

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

require(["dojo/query","dojox/NodeList/delegate"], function(query,delegate){

 query("#list").delegate("li","onclick",function(event) {
 // "this.node" is the node that was clicked
 console.log("you clicked a link!",this);
 });
})

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

通過委托, 能夠體會(huì)到使用事件委托對(duì)于開發(fā)帶來的幾個(gè)好處:

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

2.可以方便地動(dòng)態(tài)添加和修改元素,不需要因?yàn)樵氐母膭?dòng)而修改事件綁定。

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

在JavaScript編程中使用代理
上面介紹的是對(duì)DOM事件處理時(shí),利用瀏覽器冒泡機(jī)制為DOM元素添加事件代理。其實(shí)在純JS編程中,我們也可以使用這樣的編程模式,來創(chuàng)建代理對(duì)象來操作目標(biāo)對(duì)象.

var delegate = function(client, clientMethod) {
  return function() {
   return clientMethod.apply(client, arguments);
  }
 }
 var Apple= function() {
  var _color = "red";
  return {
   getColor: function() {
    console.log("Color: " + _color);
   },
   setColor: function(color) {
    _color = color;
   }
  };
 };

 var a = new Apple();
 var b = new Apple();
 a.getColor();
 a.setColor("green");
 a.getColor();
 //調(diào)用代理
 var d = delegate(a, a.setColor);
 d("blue");
 //執(zhí)行代理
 a.getColor();
 //b.getColor();


上面的例子中,通過調(diào)用delegate()函數(shù)創(chuàng)建的代理函數(shù)d來操作對(duì)a的修改。

這種方式盡管是使用了apply(call也可以)來實(shí)現(xiàn)了調(diào)用對(duì)象的轉(zhuǎn)移,但是從編程模式上實(shí)現(xiàn)了對(duì)某些對(duì)象的隱藏,可以保護(hù)這些對(duì)象不被隨便訪問和修改。

在很多框架中都引用了委托這個(gè)概念用來指定方法的運(yùn)行作用域。

比較典型的如dojo.hitch(scope,method)和ExtJS的createDelegate(obj,args)。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家學(xué)習(xí)javascript程序設(shè)計(jì)有所幫助。

相關(guān)文章

  • Javascript中的 “&” 和 “|” 詳解

    Javascript中的 “&” 和 “|” 詳解

    本文主要介紹了Javascript中的 “&” 和 “|” 的相關(guān)知識(shí)。具有很好的參考價(jià)值,下面跟著小編一起來看下吧
    2017-02-02
  • Qt6基于Qml的文件對(duì)話框演示效果

    Qt6基于Qml的文件對(duì)話框演示效果

    這篇文章主要介紹了Qt6基于Qml的文件對(duì)話框演示,包括打開單個(gè)文件配置和打開多個(gè)文件配置及保存文件配置的方法,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2022-10-10
  • ECharts設(shè)置x軸刻度間隔的2種解決方法

    ECharts設(shè)置x軸刻度間隔的2種解決方法

    在初步接觸Echarts時(shí),經(jīng)常遇到設(shè)置x軸刻度間隔的問題,這篇文章主要給大家介紹了關(guān)于ECharts設(shè)置x軸刻度間隔的2種解決方法,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-08-08
  • 純?cè)鷍s實(shí)現(xiàn)table表格的增刪

    純?cè)鷍s實(shí)現(xiàn)table表格的增刪

    本文主要介紹了純?cè)鷍avascript實(shí)現(xiàn)table表格的增刪的方法,文章底部提供了完整的代碼。需要的朋友一起來看下吧
    2017-01-01
  • 徹底理解JavaScript的原型與原型鏈

    徹底理解JavaScript的原型與原型鏈

    原型和原型鏈?zhǔn)莏s中的難點(diǎn)也是重點(diǎn),明白了原型和原型鏈會(huì)讓我們?cè)诤竺娌还苁菍W(xué)習(xí)還是工作都會(huì)更加高效,這篇文章主要給大家介紹了關(guān)于JavaScript原型與原型鏈的相關(guān)資料,需要的朋友可以參考下
    2021-10-10
  • JScript的條件編譯

    JScript的條件編譯

    JScript的條件編譯...
    2007-05-05
  • js實(shí)現(xiàn)表格單列按字母排序

    js實(shí)現(xiàn)表格單列按字母排序

    這篇文章主要為大家詳細(xì)介紹了js實(shí)現(xiàn)表格單列按字母排序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-08-08
  • input?獲取光標(biāo)位置設(shè)置光標(biāo)位置方案

    input?獲取光標(biāo)位置設(shè)置光標(biāo)位置方案

    這篇文章主要為大家介紹了input?獲取光標(biāo)位置設(shè)置光標(biāo)位置方案詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-06-06
  • 在js代碼拼接dom對(duì)象到頁面上去的模板總結(jié)(必看)

    在js代碼拼接dom對(duì)象到頁面上去的模板總結(jié)(必看)

    下面小編就為大家?guī)硪黄趈s代碼拼接dom對(duì)象到頁面上去的模板總結(jié)(必看)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-02-02
  • JS實(shí)用案例之輸入智能提示(打字機(jī)輸出效果)

    JS實(shí)用案例之輸入智能提示(打字機(jī)輸出效果)

    最近在項(xiàng)目需求中遇到之前沒有做過的功能,下面這篇文章主要給大家介紹了關(guān)于JS實(shí)用案例之輸入智能提示,文中通過實(shí)例代碼介紹的非常詳細(xì),打字機(jī)輸出效果的相關(guān)資料,需要的朋友可以參考下
    2023-01-01

最新評(píng)論