成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

jQuery 源碼系列(十一)event 總體概述

liujs / 608人閱讀

摘要:而事件委托的概念事件目標(biāo)自身不處理事件,而是將其委托給父元素或祖先元素或根元素,而借助事件的冒泡性質(zhì)由內(nèi)向外來(lái)達(dá)到最終處理事件。而且一旦出現(xiàn),局部刷新導(dǎo)致重新綁定事件。函數(shù)的用法,代表要移除的事件,表示選擇的,表示事件處理函數(shù)。

歡迎來(lái)我的專欄查看系列文章。

這次的內(nèi)容是來(lái)介紹關(guān)于 jQuery 的事件委托。不過(guò)在之前呢有必要先來(lái)了解一下 JS 中的事件委托與冒泡,我之前也寫(xiě)過(guò)類似的博客,事件冒泡與捕獲

由 JS 事件引入

事件是 JS DOM 中極具活力的內(nèi)容,你可以隨時(shí)監(jiān)聽(tīng) DOM 的變化,并對(duì)它們及時(shí)的做出反應(yīng),如果你不是太懂 JS 中的事件,建議你先去看一些相關(guān)介紹的文章,直接看 jQuery 中的事件委托頭會(huì)頭大的。

事件的處理順序由兩個(gè)環(huán)節(jié),一個(gè)是捕獲環(huán)節(jié),一個(gè)是冒泡環(huán)節(jié),借用別人的一張圖:

如果把處理也算進(jìn)的話,整個(gè)事件分為三個(gè)階段,分別是捕獲階段,目標(biāo)處理階段和冒泡階段。捕獲階段由外向內(nèi)尋找 target,冒泡階段由內(nèi)向外直到根結(jié)點(diǎn)。這只是一個(gè)事件,當(dāng)這三個(gè)階段中又穿插著更多的事件時(shí),還需要將事件的執(zhí)行順序考慮進(jìn)去。

而 jQuery 事件委托的概念:事件目標(biāo)自身不處理事件,而是將其委托給父元素或祖先元素或根元素,而借助事件的冒泡性質(zhì)(由內(nèi)向外)來(lái)達(dá)到最終處理事件。

jQuery 中的事件優(yōu)化

首先必須要知道,綁定事件越多,瀏覽器內(nèi)存占用越大,就會(huì)間接的影響性能。而且一旦出現(xiàn) ajax,局部刷新導(dǎo)致重新綁定事件。

使用事件委托可以解決以上帶來(lái)的問(wèn)題,借助事件的冒泡,尤其當(dāng)一個(gè)父元素的子元素過(guò)多,而且子元素綁定的事件非常多時(shí),委托事件的作用就體現(xiàn)出來(lái)了。

我本人不善于比較 JS 中的性能問(wèn)題,感興趣的可以去看看這篇文章關(guān)于事件委托性能的設(shè)計(jì)和比較。深入理解-事件委托。

在早期的 jQuery 版本,使用的是 .delegate().bind()、.live()等方法來(lái)實(shí)現(xiàn)事件監(jiān)聽(tīng),當(dāng)然也包括.click()方法,隨著 jQuery 的發(fā)展,像 live 方法已經(jīng)明確從 jQuery 中刪除,而其余的方法,比如 bind 方法也將在 3.0 之后的版本陸續(xù)刪除,取而代之的是 .on()方法。而且剩下的其它方法都是通過(guò) on 方法來(lái)間接實(shí)現(xiàn)的,如果介紹,只需要看 on 的源碼即可。

on 函數(shù)在 jQuery 中的用法也很簡(jiǎn)單,.on( events [, selector ] [, data ], handler(eventObject) )events 表示綁定的事件,比如 "click" 或 "click mouseleave",selector 和 data 是可選的,分別表示要綁定事件的元素和要執(zhí)行的數(shù)據(jù),handler 表示事件執(zhí)行函數(shù)。

off 函數(shù)的用法 .off( events [, selector ] [, handler ] ),events 代表要移除的事件,selector 表示選擇的 dom,handler 表示事件處理函數(shù)。還有更殘暴的比如 .off()不接受任何參數(shù),表示著移除所有 on 綁定的函數(shù)。

on off 函數(shù)源碼

雖然我分析的源碼時(shí) jQuery 3.1.1,但這個(gè)時(shí)候 bind 和 delegate 函數(shù)并沒(méi)有從源碼中移除呢,先來(lái)看看它們?cè)趺凑{(diào)用 on:

jQuery.fn.extend( {
  bind: function( types, data, fn ) {
    return this.on( types, null, data, fn );
  },
  unbind: function( types, fn ) {
    return this.off( types, null, fn );
  },
  delegate: function( selector, types, data, fn ) {
    return this.on( types, selector, data, fn );
  },
  undelegate: function( selector, types, fn ) {
    // ( namespace ) or ( selector, types [, fn] )
    return arguments.length === 1 ?
      this.off( selector, "**" ) :
      this.off( types, selector || "**", fn );
  }
} );

可以看得出來(lái),全都被 on 和 off 這兩個(gè)函數(shù)來(lái)處理了。

jQuery.fn.extend( {
  on: function (types, selector, data, fn) {
    // on 又依托于全局的 on 函數(shù)
    return on(this, types, selector, data, fn);
  }
} );
function on( elem, types, selector, data, fn, one ) {
  var origFn, type;

  // 支持 object 的情況
  if ( typeof types === "object" ) {

    // ( types-Object, selector, data )
    if ( typeof selector !== "string" ) {

      // ( types-Object, data )
      data = data || selector;
      selector = undefined;
    }
    // 一次執(zhí)行 object 的每一個(gè)
    for ( type in types ) {
      on( elem, type, selector, data, types[ type ], one );
    }
    return elem;
  }
  // 參數(shù)為兩個(gè)的情況
  if ( data == null && fn == null ) {

    // ( types, fn )
    fn = selector;
    data = selector = undefined;
  } else if ( fn == null ) {
    if ( typeof selector === "string" ) {

      // ( types, selector, fn )
      fn = data;
      data = undefined;
    } else {

      // ( types, data, fn )
      fn = data;
      data = selector;
      selector = undefined;
    }
  }
  if ( fn === false ) {
    // returnFalse 是一個(gè)返回 false 的函數(shù)
    fn = returnFalse;
  } else if ( !fn ) {
    return elem;
  }

  if ( one === 1 ) {
    origFn = fn;
    fn = function( event ) {

      // Can use an empty set, since event contains the info
      jQuery().off( event );
      return origFn.apply( this, arguments );
    };

    // Use same guid so caller can remove using origFn
    fn.guid = origFn.guid || ( origFn.guid = jQuery.guid++ );
  }
  return elem.each( function() {
    // 關(guān)鍵
    jQuery.event.add( this, types, fn, data, selector );
  } );
}

是的,你沒(méi)有看錯(cuò),這個(gè)全局的 on 函數(shù),其實(shí)只是起到了校正參數(shù)的作用,而真正的大頭是:

jQuery.event = {
  global = {},
  add: function(){...},
  remove: function(){...},
  dispatch: function(){...},
  handlers: function(){...},
  addProp: function(){...},
  fix: function(){...},
  special: function(){...}
}

off 函數(shù):

jQuery.fn.off = function (types, selector, fn) {
  var handleObj, type;
  if (types && types.preventDefault && types.handleObj) {
    // ( event )  dispatched jQuery.Event
    handleObj = types.handleObj;
    jQuery(types.delegateTarget).off(
      handleObj.namespace ? handleObj.origType + "." + handleObj.namespace : handleObj.origType,
      handleObj.selector,
      handleObj.handler
    );
    return this;
  }
  if (typeof types === "object") {
    // ( types-object [, selector] )
    for (type in types) {
      this.off(type, selector, types[type]);
    }
    return this;
  }
  if (selector === false || typeof selector === "function") {
    // ( types [, fn] )
    fn = selector;
    selector = undefined;
  }
  if (fn === false) {
    fn = returnFalse;
  }
  return this.each(function() {
    // 關(guān)鍵
    jQuery.event.remove(this, types, fn, selector);
  });
}
總結(jié)

可見(jiàn) jQuery 對(duì)于參數(shù)的放縱導(dǎo)致其處理起來(lái)非常復(fù)雜,不過(guò)對(duì)于使用者來(lái)說(shuō),卻非常大便利。

委托事件也帶來(lái)了一些不足,比如一些事件無(wú)法冒泡,load、submit 等,會(huì)加大管理等復(fù)雜,不好模擬用戶觸發(fā)事件等。

參考

jQuery 2.0.3 源碼分析 事件綁定 - bind/live/delegate/on
深入理解-事件委托
.on()

本文在 github 上的源碼地址,歡迎來(lái) star。

歡迎來(lái)我的博客交流。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/81780.html

相關(guān)文章

  • JavaScript 工作原理之十一-渲染引擎及性能優(yōu)化小技巧

    摘要:在中渲染樹(shù)中的每個(gè)節(jié)點(diǎn)即是一個(gè)渲染器或者渲染器對(duì)象。計(jì)算的樣式每個(gè)渲染器對(duì)象代表一個(gè)矩形區(qū)域通常是和一個(gè)節(jié)點(diǎn)的盒模型相對(duì)應(yīng)。坐標(biāo)系統(tǒng)是相對(duì)于根渲染器的。根渲染器的定位為和大小即為瀏覽器窗口的可視化部分比如。渲染器作廢其在屏幕上的矩形區(qū)域。 原文請(qǐng)查閱這里,略有刪減,本文采用知識(shí)共享署名 4.0 國(guó)際許可協(xié)議共享,BY Troland。 本系列持續(xù)更新中,Github 地址請(qǐng)查閱這里。 ...

    GraphQuery 評(píng)論0 收藏0
  • JavaScript 工作原理之十一-渲染引擎及性能優(yōu)化小技巧

    摘要:在中渲染樹(shù)中的每個(gè)節(jié)點(diǎn)即是一個(gè)渲染器或者渲染器對(duì)象。計(jì)算的樣式每個(gè)渲染器對(duì)象代表一個(gè)矩形區(qū)域通常是和一個(gè)節(jié)點(diǎn)的盒模型相對(duì)應(yīng)。坐標(biāo)系統(tǒng)是相對(duì)于根渲染器的。根渲染器的定位為和大小即為瀏覽器窗口的可視化部分比如。渲染器作廢其在屏幕上的矩形區(qū)域。 原文請(qǐng)查閱這里,略有刪減,本文采用知識(shí)共享署名 4.0 國(guó)際許可協(xié)議共享,BY Troland。 本系列持續(xù)更新中,Github 地址請(qǐng)查閱這里。 ...

    Allen 評(píng)論0 收藏0
  • JavaScript 工作原理之十一-渲染引擎及性能優(yōu)化小技巧

    摘要:在中渲染樹(shù)中的每個(gè)節(jié)點(diǎn)即是一個(gè)渲染器或者渲染器對(duì)象。計(jì)算的樣式每個(gè)渲染器對(duì)象代表一個(gè)矩形區(qū)域通常是和一個(gè)節(jié)點(diǎn)的盒模型相對(duì)應(yīng)。坐標(biāo)系統(tǒng)是相對(duì)于根渲染器的。根渲染器的定位為和大小即為瀏覽器窗口的可視化部分比如。渲染器作廢其在屏幕上的矩形區(qū)域。 原文請(qǐng)查閱這里,略有刪減,本文采用知識(shí)共享署名 4.0 國(guó)際許可協(xié)議共享,BY Troland。 本系列持續(xù)更新中,Github 地址請(qǐng)查閱這里。 ...

    RyanQ 評(píng)論0 收藏0
  • jQuery 源碼系列(一)總體架構(gòu)

    摘要:到目前為止,的貢獻(xiàn)者團(tuán)隊(duì)共名成員,多條,可想而知,是一個(gè)多么龐大的項(xiàng)目。參考源碼分析整體架構(gòu)源碼解析讀書(shū)筆記第二章構(gòu)造對(duì)象函數(shù)詳解本文在上的源碼地址,歡迎來(lái)。 歡迎來(lái)我的專欄查看系列文章。 決定你走多遠(yuǎn)的是基礎(chǔ),jQuery 源碼分析,向長(zhǎng)者膜拜! 我雖然接觸 jQuery 很久了,但也只是局限于表面使用的層次,碰到一些問(wèn)題,找到 jQuery 的解決辦法,然后使用。顯然,這種做法的...

    svtter 評(píng)論0 收藏0
  • 前端基礎(chǔ)進(jìn)階(十一):詳細(xì)圖解jQuery對(duì)象,以及如何擴(kuò)展jQuery插件

    摘要:而在構(gòu)造函數(shù)中,返回了的實(shí)例對(duì)象。在中直接返回過(guò)的實(shí)例,這里的是的真正構(gòu)造函數(shù)最后對(duì)外暴露入口時(shí),將字符與對(duì)等起來(lái)。因此當(dāng)我們直接使用創(chuàng)建一個(gè)對(duì)象時(shí),實(shí)際上是創(chuàng)建了一個(gè)的實(shí)例,這里的正真構(gòu)造函數(shù)是原型中的方法。 showImg(https://segmentfault.com/img/remote/1460000008749398); 早幾年學(xué)習(xí)前端,大家都非常熱衷于研究jQuery源...

    RebeccaZhong 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<