JavaScript事件代理和委託

瀏覽器的事件冒泡

當事件發生後,這個事件就要開始傳播。例如咱們點擊一個按鈕時,就會產生一個click事件,但這個按鈕自己不能處理這個事件,事件必須從這個按鈕傳播出去,從而到達可以處理這個事件的代碼中。從裏到外,直至它被處理,或者它到達了對象層次的最頂層,即document對象(有些瀏覽器是window)javascript

(1)冒泡型事件:事件按照從最特定的事件目標到最不特定的事件目標(document對象)的順序觸發。html

  IE 5.5: div -> body -> documentjava

  IE 6.0: div -> body -> html -> documentnode

  Mozilla 1.0: div -> body -> html -> document -> windowjquery

(2)捕獲型事件(event capturing):事件從最不精確的對象(document 對象)開始觸發,而後到最精確的對象web

(3)DOM事件流:同時支持兩種事件模型:捕獲型事件和冒泡型事件,可是,捕獲型事件先發生。兩種事件流會觸及DOM中的全部對象,從document對象開始,也在document對象結束。瀏覽器

DOM2.0模型將事件處理流程分爲三個階段:1、事件捕獲階段,2、事件目標階段,3、事件起泡階段。如圖:函數

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

事件目標:當到達目標元素以後,執行目標元素該事件相應的處理函數。若是沒有綁定監聽函數,那就不執行。spa

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

每一個 event 都有一個event.bubbles屬性,能夠知道它能否冒泡。

JS事件代理

對於須要給不少列表式元素添加事件時,能夠直接將事件添加到它們的父節點,將事件委託給父節點來觸發處理函數。利用瀏覽器的事件冒泡機制實現單個元素事件處理

假設有一個ul 的父節點包含一些列li的子節點,但願實現的效果是:鼠標移到li上時彈出懸浮窗,即li須要出發處理事件,一般寫法是給每一個li綁定onclick等監聽事件。

//html:
<ul id="parent">
  <li id="child1">Item 1</li>
  <li id="child2">Item 2</li>
  <li id="child3">Item 3</li>
  <li id="child4">Item 4</li>
  <li id="child5">Item 5</li>
</ul>

js代碼:

//js:
function addListeners(li_child){
    li_child = function clickHandler(){...};
}

window.onload = function(){
    var ul_parent = document.getElementById("parent");
    var li_childs = ulNode.getElementByTagName("li");
    for(var i=0, l = li_childs; i < l; i++){
        addListener(li_childs[i]);
    }   
}

若是這個ul中的li子元素會頻繁地添加或者刪除,咱們就須要在每次添加li的時候都調用這個addListener方法來爲每一個li節點添加事件處理函數。這就添加的複雜度和出錯的可能性。

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

// 獲取父節點,併爲它添加一個click事件
document.getElementById("parent").addEventListener("click",function(e) {
  // 檢查事件源e.targe是否爲Li
  if(e.target && e.target.nodeName.toUpperCase == "li") {
    // 真正的處理過程在這裏
    ...
  }
});

爲父節點添加一個click事件,當子節點被點擊的時候,click事件會從子節點開始向上冒泡。父節點捕獲到事件以後,經過判斷e.target.nodeName來判斷是否爲咱們須要處理的節點。而且經過e.target拿到了被點擊的Li節點。從而能夠獲取到相應的信息,並做處理。

jQuery中delegate函數

delegate() 方法爲指定的元素(屬於被選元素的子元素)添加一個或多個事件處理程序,並規定當這些事件發生時運行的函數,使用 delegate() 方法的事件處理程序適用於當前或將來的元素(好比由腳本建立的新元素)。

$("#parent").delegate("li", "click", function(){
  // "$(this)" is the node that was clicked
  ...
});

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

優勢

經過上面的介紹,你們應該可以體會到使用事件委託對於web應用程序帶來的幾個優勢:

1.管理的函數變少了。不須要爲每一個元素都添加監聽函數。對於同一個父節點下面相似的子元素,能夠經過委託給父元素的監聽函數來處理事件。

2.能夠方便地動態添加和修改元素,不須要由於元素的改動而修改事件綁定。

3.JavaScript和DOM節點之間的關聯變少了,這樣也就減小了因循環引用而帶來的內存泄漏發生的機率。

jquery中對冒泡和默認行爲的阻止方法

網頁中的某些元素是有本身的默認行爲的,好比果超連接單擊後須要跳轉,提交按鈕點擊後須要提交表單,有時須要阻止這些行爲,也就是默認行爲,jQuery對這個問題進行了必要的擴展和封裝

 $("element").bind("click",function(event){  });   

 //event爲事件對象event.stopPropagation();   //中止事件冒泡  

jquery中可用用preventDefault()的方法來阻止元素的默認行爲

$('#submit').bind('click', function(event) {

    var username = $('#username').val();

    if (username == "") {

        alert('用戶名不能爲空!');

        event.preventDefault(); //阻止默認行爲  

    }

})

jquery中對冒泡和默認行爲的阻止方法能夠改寫,改寫後可以達到一樣的效果

event.preventDefault();  改寫爲:  return false;
event.stopPropagation();  改寫爲:  return false;
相關文章
相關標籤/搜索