淺談 jQuery 事件源碼定位問題
昨天群里有人問了個(gè)事件源碼定位的問題,簡單描述下是這樣的。
在一個(gè)不是自己寫的頁面上,如何快速定位到他綁定的事件代碼在哪?(頁面用的是jQuery)
這個(gè)問題,說難不難,說簡單也沒那么簡單,萬一用的是委托之類也會(huì)麻煩點(diǎn)。
在 chrome 的控制臺(tái)里有個(gè) Event Listeners,這里會(huì)顯示你所選擇元素的事件,如果是原生事件,他會(huì)直接顯示,
你點(diǎn)擊一下事件就會(huì)跳到對(duì)應(yīng)代碼里了,可是 jQuery 綁定的事件卻不是這樣的,你點(diǎn)擊后只會(huì)跳到 jQuery 源碼里,
min后的jQuery源碼密密麻麻的,看著都眼花。
關(guān)于jQuery對(duì)于事件的管理,大牛們也分析的非常透徹了,我就不啰嗦了,因?yàn)椴皇俏覀兘裉煲f的重點(diǎn)。
我們要說的重點(diǎn)是怎么定位到事件源碼處。因?yàn)閖Query版本眾多,而且重構(gòu)過多次,所以要分情況來說了。
基本上 1.2.6-1.8 和 1.9 兩種情況,經(jīng)過測(cè)試,大體上定為下面2個(gè)版本
1.2.6-1.8 用 $.data( elem, "events", undefined, true );
1.9+ 用 $._data( elem, "events" );
PS: 你現(xiàn)在也可以按 F12 打開控制臺(tái)看看結(jié)果,當(dāng)然也可以復(fù)制下面的源碼自己測(cè)試。
由于谷歌被墻的厲害,所以把cdn換成百度的了。2014-06-07
<!doctype html> <html lang="en"> <head> <meta charset="UTF-8"> <title>test</title> <script src="http://libs.baidu.com/jquery/1.4.0/jquery.js"></script> </head> <body> <input type="button" id="testbtn" value="testbtn" /> <script> var version = ["1.2.6", "1.3.0", "1.4.0", "1.5.0", "1.6.0", "1.7.0", "1.8.0", "1.9.0", "1.10.0"], elem = $("#testbtn")[0], // 待操作的元素 url, // jquery 地址 jq = null, // 保存新的jquery句柄 jqver, // jqury 版本 fn; // 函數(shù)句柄 for (var i = 0; i < version.length; i++) { url = "http://libs.baidu.com/jquery/" + version[i] + "/jquery.min.js"; $.getScript(url, function() { jq = $.noConflict(true); // 釋放控制權(quán) jqver = jq.fn.jquery; // 當(dāng)前 jquery 版本 fn = new Function('ver_' + jqver.replace(/\./g, "_"), ''); // 生成類似 function (ver_1_9_0) {} 這樣的函數(shù) jq(elem).click(fn).click(fn).bind("test", fn); // 普通事件和自定義事件 console.log( jqver, jq.data && jq.data(elem, "events", undefined, true), jq._data && jq._data(elem, "events") ); }); } </script> </body> </html>
如果不出意外,你可以在控制臺(tái)看到這樣的顯示結(jié)果
展開后可以看到綁定的函數(shù)參數(shù)里的版本和當(dāng)前版本是對(duì)應(yīng)的。
可以看到
1.2.6-1.4 只支持 $.data( elem, "events", undefined, true );
1.5-1.8 兩者都支持
1.9-1.11 只支持 $._data( elem, "events" );
那么我們可以寫個(gè)函數(shù)簡單的兼容下,然他全兼容即可
function lookEvents (elem) { return $.data ? $.data( elem, "events", undefined, true ) : $._data( elem, "events" ); }
現(xiàn)在調(diào)用 lookEvents 就可以得到對(duì)應(yīng)的 events 對(duì)象了。
雖然可以看到了我們綁定的自定義事件,但還是不知道他在哪個(gè)文件哪一行啊。
下面我們就來定位他的具體位置,我們就拿 1.7 的試試。
PS: 下面操作都是在控制臺(tái)完成,我的環(huán)境是 chrome 34
function lookEvents (elem) { return $.data ? $.data( elem, "events", undefined, true ) : $._data( elem, "events" ); } var event = lookEvents($("#testbtn")[0]); // 獲取綁定的事件 event.click[0].handler // 獲取click事件的第一個(gè)事件源碼地址
復(fù)制到控制臺(tái),按回車運(yùn)行后,不出意外可以看到下面這個(gè)結(jié)果。
有沒有看到右下角的 1.html:36 這個(gè)就是源碼所在的文件和對(duì)應(yīng)的行號(hào)了。
你可以直接點(diǎn)擊 1.html:36 跳到對(duì)應(yīng)的代碼處,是不是覺得很給力啊。
上面方法適用于 1.5+ 版本的 jQuery,對(duì)于 1.2.6-1.4 的版本,稍微有點(diǎn)不同,不過也非常簡單。
function lookEvents (elem) { return $.data ? $.data( elem, "events", undefined, true ) : $._data( elem, "events" );}var event = lookEvents($("#testbtn")[0]); // 獲取綁定的事件event.click; // 查看有幾個(gè)click事件,如果要查看其他事件直接輸入 event 然后回車即可
上面看到的編碼就是對(duì)應(yīng)事件句柄了,比如我這的 1,2 事件(如下圖顯示), 這個(gè)編號(hào)不是按順序的,這個(gè)要注意。
event.click[1] // 獲取click事件的 id是1 的事件源碼地址
不出意外可以看到下面這個(gè)結(jié)果。
從操作來說,不管是 1.2.6-1.4 還是 1.5+ 版本 都差不多,只是 1.5+ 利用數(shù)組模式管理函數(shù)句柄了,比較方便。
好了,該說的都說完了,小伙伴們各種測(cè)試起來吧。
- jQuery異步加載數(shù)據(jù)并添加事件示例
- jQuery響應(yīng)鼠標(biāo)事件并隱藏與顯示input默認(rèn)值
- jquery bind(click)傳參讓列表中每行綁定一個(gè)事件
- JQuery實(shí)現(xiàn)表格動(dòng)態(tài)增加行并對(duì)新行添加事件
- jquery阻止后續(xù)事件只執(zhí)行第一個(gè)事件
- jquery中的常用事件bind、hover、toggle等示例介紹
- jQuery新的事件綁定機(jī)制on()示例應(yīng)用
- JQuery1.8 判斷元素是否綁定事件的方法
- jQuery學(xué)習(xí)總結(jié)之jQuery事件
- jquery 為a標(biāo)簽綁定click事件示例代碼
- jquery 新建的元素事件綁定問題解決方案
- jQuery移除元素自動(dòng)解綁事件實(shí)現(xiàn)思路及代碼
- jQuery事件用法實(shí)例匯總
相關(guān)文章
jquery 實(shí)現(xiàn)回車登錄詳解及實(shí)例代碼
這篇文章主要介紹了jquery 實(shí)現(xiàn)回車登錄詳解的相關(guān)資料,整理了幾種方法及簡單實(shí)現(xiàn)實(shí)例,需要的朋友可以參考下2016-10-10Jquery多選框互相內(nèi)容交換的實(shí)例代碼
這篇文章介紹了Jquery多選框互相內(nèi)容交換的實(shí)例代碼,有需要的朋友可以參考一下2013-07-07jQuery、layer實(shí)現(xiàn)彈出層的打開、關(guān)閉功能
這篇文章主要介紹了jQuery、layer實(shí)現(xiàn)彈出層的打開、關(guān)閉功能,需要的朋友可以參考下2017-06-06