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

資訊專欄INFORMATION COLUMN

jQuery之事件綁定到觸發(fā)全過程及知識點補充

Jioby / 519人閱讀

摘要:十的觸發(fā)機制被點擊了元素本身綁定了一個事件,但是是原生事件,它是靠綁定來觸發(fā)事件的。

前言:
最重要的還是最后的流程圖,可以試著根據(jù)流程圖手寫實現(xiàn)$().on(),下篇文章會放出模擬實現(xiàn)的代碼。

一、舉例

這是A
這是C
$("#A").on("click" ,function (event) { console.log(event,"A被點擊了") }) $("#A").on("click" ,"#C",function (event) { console.log(event,"點擊了C,即C委托A的click事件被點擊了") })

二、$().on()
(1)進行參數(shù)的調(diào)整
(2)調(diào)用jQuery.event.add()方法

三、jQuery.event.add()最終調(diào)用elem.addEventListener()來綁定事件
注意:
(1)綁定常用的事件(如:click、focus),使用handleObj保存

handleObj = jQuery.extend( {
          //click,mouseout...
          type: type,
          //click,mouseout...
          origType: origType,
          data: data,
          //事件處理函數(shù),如 function(){console.log("aaaa")}
          handler: handler,
          //索引,用于關(guān)聯(lián)元素和事件
          guid: handler.guid,
          //事件委托的標(biāo)志,也是委托的對象選擇器
          selector: selector,
          needsContext: selector && jQuery.expr.match.needsContext.test( selector ),
          //命名空間,同一click事件有兩個事件處理程序handler的話,
          //用這個標(biāo)識,方便刪除或添加handler
          namespace: namespaces.join( "." )
        }, handleObjIn );

(2)如果綁定的是自定義事件(如:windowResize),則使用handleObjIn保存

if ( handler.handler ) {
   handleObjIn = handler;
   handler = handleObjIn.handler;
   selector = handleObjIn.selector;
}

(1)、(2)都會初始化事件處理器(addEventListener):

        //第一次綁定事件,走這里
        // Init the event handler queue if we"re the first
        if ( !( handlers = events[ type ] ) ) {
          handlers = events[ type ] = [];
          handlers.delegateCount = 0;
          // Only use addEventListener if the special events handler returns false
          if ( !special.setup ||
            special.setup.call( elem, data, namespaces, eventHandle ) === false ) {
            //目標(biāo)元素有addEventListener的話,調(diào)用綁定click事件
            //eventHandle就綁定到addEventListener上
            if ( elem.addEventListener ) {
              elem.addEventListener( type, eventHandle );
            }
          }
        }

四、jQuery的事件綁定為何不直接綁定在目標(biāo)元素身上,而是元素和事件分離?
打印$("#A")

console.log($("#A"),"aaaaaa46")

不要在意jQueryId不同的問題,每次刷新網(wǎng)頁它都會變化

可以看到

jQuery的事件和觸發(fā)事件的handler是分離的,
事件集合 存在 事件緩存dataPrivevents上,

//獲取數(shù)據(jù)緩存
elemData = dataPriv.get( elem );

而handler是由jQuery.event.dispatch()處理

elemData.handle = function( e ) {
  jQuery.event.dispatch.apply( elem, arguments )
}

為什么要分離?因為元素如果綁定click事件一百次,很耗內(nèi)存。所以需要將這一百個同類型的事件保存到一個click事件集合中,然后在這一大個click事件集合內(nèi),根據(jù)guid來執(zhí)行某一次的click處理代碼

同一事件的處理:

$("body").on("click", "#one", function(e) {
  show("委托到one觸發(fā)")
})
$("body").on("click", "#two", function(e) {
  show("委托到two觸發(fā)")
})

events是jQuery內(nèi)部的事件隊列
handle是真正綁定到element上的事件處理函數(shù)

body:{
  events:{
    click:[
      0:{
        guid: 1,
        data: undefined,
        namespace: "",
        origType: "click",
        //事件委托的標(biāo)志
        selector: "#one",
        type: "click",
        handler: function(){xxx},
        }
      1:{
        guid: 2,
        data: undefined,
        namespace: "",
        origType: "click",
        //事件委托的標(biāo)志
        selector: "#two",
        type: "click",
        handler: function(){xxx},
        }
    ]
  },
  handle: function(){
    jQuery.event.dispatch.apply( elem, arguments )
  }
}

可以看到,針對同一類型的事件(如click),重復(fù)綁定不會再創(chuàng)建新的內(nèi)存(new Object會有新內(nèi)存),而是在events里添加新的綁定事件。

記得看第十一點!

五、guid的作用?
添加guid的目的是因為handler沒有直接跟元素節(jié)點發(fā)生關(guān)聯(lián),所以需要一個索引來尋找或者刪除handler

六、命名空間namespace的作用?

 $("#one").on("click.one",function () {
   console.log("one被點擊了")
 })
 $("#one").on("click.two",function () {
   console.log("two被點擊了")
 })

命名空間為:

#one:{
  events:{
    click:[
      0:{
        namespace: "one",
        handler: function(){console.log("one被點擊了")},
        }
      1:{
        namespace: "two",
        handler: function(){xxx},
        }
    ]
  },
}

利用命名空間刪除事件:

$("#one").off("click.one")

七、jQuery.event.special 的處理機制
綁定的事件,有些是不能統(tǒng)一處理的,比如load事件,是不支持冒泡的,所以即使開發(fā)者未用event.stopPropagation,jQuery也要阻止其冒泡:

jQuery.event ={
  special: {
    load: {
        // Prevent triggered image.load events from bubbling to window.load
        //阻止冒泡
        noBubble: true
    },
    focus: {
        // Fire native event if possible so blur/focus sequence is correct
        trigger: function() { },
        delegateType: "focusin"
    },
  }
}

八、外部是Event,內(nèi)部是數(shù)據(jù)緩存events,兩者是不一樣的
外部Event:

$().on("click","#B",function(event){
  console.log("A被點擊了")
})

//click的event就是jQuery.Event
jQuery.Event{
  handleObj{
    data:undefined,
    guid: 2,
    handler:function(){console.log("A被點擊了")},
    namespace: "clickA",
    origType: "click",
    selector: "#B",
    type: "click.clickA",
  },
  originalEvent:{
    //就是MouseEvent
  },
  target:div#B,
  type: "click",
  delegateTarget: div#A,
  //fix 的標(biāo)志
  jQuery331087940272164138: true,
  currentTarget: div#A,
  isDefaultPrevented:xxx,
  timeStamp:Date.now(),
  isDefaultPrevented:function(){return false}
}

內(nèi)部緩存events:

let events = dataPriv.get( this, "events" )

events[
    delegantCount:1,
   {
      data:undefined,
      guid: 2,
      handler:function(){console.log("B委托A被點擊了")},
      namespace: "clickA",
      origType: "click",
      selector: "#B",
      type: "click.clickA",
    },
   {
     data:undefined,
      guid: 1,
      handler:function(){console.log("A被點擊了")},
      namespace: "",
      origType: "click",
      selector: undefined,
      type: "click",
    }   


]

九、為什么要使用fix()來重構(gòu) js 的原生 MouseEvent 對象呢?
(1)jQuery 有自己的一套event處理機制,所以需要符合jQueryevent對象
(2)可以傳遞 data 數(shù)據(jù),即用戶自定義的數(shù)據(jù)。

十、trigger的觸發(fā)機制

  $("#A").on("click" ,function (event) {
    console.log(event,"A被點擊了")
  })

元素#A本身綁定了一個click事件,但是click是原生事件,它是靠 addEventListener綁定來觸發(fā)事件的。

但是!jQuerytrigger是能夠無差別模擬這個交互行為的

$("#A").trigger("click")

trigger()的功能上就可以解釋 為什么jQuery要設(shè)計元素與數(shù)據(jù)分離了:

如果是直接綁定的話就無法通過trigger的機制去觸發(fā)click事件,
正是因為jQuery沒有直接把事件相關(guān)的handler與元素直接綁定,而是采用了分離處理,
所以我們通過trigger觸發(fā)click事件與addEventListener觸發(fā)click事件的處理流程是一致的,不同的只是觸發(fā)的方式而已。

但是,通trigger觸發(fā)的事件是沒有事件對象(event)、冒泡(bubble)這些特性的,所以我們需要有一個功能 能模擬出事件對象,然后生成一個遍歷樹(eventPath)模擬出冒泡行為,這個就交給了trigger方法了

關(guān)于$().trigger()的源碼解析請看:jQuery源碼解析之trigger()

最后,附上自己做的 jQuery事件綁定到觸發(fā)全過程的流程圖:

(完)

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

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

相關(guān)文章

  • jQuery源碼解析click()的事件綁定

    摘要:階段二目標(biāo)瀏覽器找到監(jiān)聽器后,就運行該監(jiān)聽器階段三冒泡目標(biāo)到祖在事件自下而上到達(dá)目標(biāo)節(jié)點的過程中,瀏覽器會檢測不是針對該事件的監(jiān)聽器用來捕獲事件,并運行非捕獲事件的監(jiān)聽器。注意下這種情況,是在里的具體實現(xiàn),即調(diào)用一次后,就執(zhí)行,卸載事件。 showImg(https://segmentfault.com/img/remote/1460000019304809); 前言:這篇依舊長,請耐...

    figofuture 評論0 收藏0
  • 最近遇的前端面試題(2017.03.08更新版)

    摘要:通過管理組件通信通過驅(qū)動視圖比較差異進行更新操作作者第七頁鏈接來源知乎著作權(quán)歸作者所有,轉(zhuǎn)載請聯(lián)系作者獲得授權(quán)。達(dá)到無刷新的效果。對象的狀態(tài)不受外界影響。對象代表一個異步操作,有三種狀態(tài)進行中已完成,又稱和已失敗。 以下問題解釋非本人原創(chuàng),是根據(jù)面試經(jīng)驗整理后覺得更容易理解的解釋版本,歡迎補充。 一. 輸入url后的加載過程 從輸入 URL 到頁面加載完成的過程中都發(fā)生了什么 計算機...

    linkFly 評論0 收藏0
  • 最近遇的前端面試題(2017.03.08更新版)

    摘要:通過管理組件通信通過驅(qū)動視圖比較差異進行更新操作作者第七頁鏈接來源知乎著作權(quán)歸作者所有,轉(zhuǎn)載請聯(lián)系作者獲得授權(quán)。達(dá)到無刷新的效果。對象的狀態(tài)不受外界影響。對象代表一個異步操作,有三種狀態(tài)進行中已完成,又稱和已失敗。 以下問題解釋非本人原創(chuàng),是根據(jù)面試經(jīng)驗整理后覺得更容易理解的解釋版本,歡迎補充。 一. 輸入url后的加載過程 從輸入 URL 到頁面加載完成的過程中都發(fā)生了什么 計算機...

    Nosee 評論0 收藏0
  • 最近遇的前端面試題(2017.03.08更新版)

    摘要:通過管理組件通信通過驅(qū)動視圖比較差異進行更新操作作者第七頁鏈接來源知乎著作權(quán)歸作者所有,轉(zhuǎn)載請聯(lián)系作者獲得授權(quán)。達(dá)到無刷新的效果。對象的狀態(tài)不受外界影響。對象代表一個異步操作,有三種狀態(tài)進行中已完成,又稱和已失敗。 以下問題解釋非本人原創(chuàng),是根據(jù)面試經(jīng)驗整理后覺得更容易理解的解釋版本,歡迎補充。 一. 輸入url后的加載過程 從輸入 URL 到頁面加載完成的過程中都發(fā)生了什么 計算機...

    劉東 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<