jquery绑定原理 简单解析与实现代码分享


Posted in Javascript onSeptember 06, 2011

jq里面有一个data的方法,给dom元素绑定相关的数据的。当给dom用jq的方法绑定了事件,会生成对应的时间列表
可以看下面的例子(请在firefox中查看 因为firefox中对象支持toSource())

<!DOCTYPE html> 
<html> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title></title> 
</head> 
<body> 
<div id="test"></div> 
<script type="text/javascript" src="http://common.cnblogs.com/script/jquery.js"></script> 
<script type="text/javascript"> 
window.onload = function(){ 
alert($.data($('#test')[0],'events'));//null 
alert($.data($('#test')[0],'handle'));//null 
$('#test') 
.bind('click',function(){ 
alert(1) 
}) 
.bind('mouseover',function(){ 
alert(2) 
}) 
.bind('click',function(){ 
alert(3) 
}) 
.bind('click',function(){ 
alert(4) 
}) 
alert($.data($('#test')[0],'events').toSource());//时间列表 
alert($.data($('#test')[0],'handle').toSource());//执行的函数 
} 
</script> 
</body> 
</html>

data是给元素绑定数据的
数据源是 cache对象
当元素绑定数据的时候 会给元素添加一个属性 jQueryxxx xxx为执行jq的时间戳
这里要说明一下,有一个uuid 他是累加的
jQueryxxx的值就是这个uuid
cache 的 key就是这个 uuid
value就是要存的数据
data对于事件的绑定是很重要的................................
function now(){ 
return +new Date; 
}; 
var win = this, 
expando = "jQuery" + now(), 
uuid = 0, 
cache = {}; 
win.data = function(elem, name, data){ 
var id = elem[expando]; 
if(!id) 
id = elem[expando] = ++uuid; 
if(name&&!cache[id]) 
cache[id] = {}; 
if(data !== undefined) 
cache[id][name] = data; 
return name 
? cache[id][name] 
: id; 
} 
win.removeData = function(elem, name){ 
var id = elem[expando]; 
if (name){ 
if (cache[id]) { 
delete cache[id][name]; 
name = ""; 
for ( name in cache[ id ] ) 
break; 
if ( !name ) 
removeData(elem); 
} 
}else{ 
try { 
delete elem[expando]; 
} catch(e){ 
if ( elem.removeAttribute ) 
elem.removeAttribute( expando ); 
} 
delete cache[id]; 
} 
} 
win.each = function( object, callback, args ) { 
var name, i = 0, length = object.length; 
if ( args ) { 
if ( length === undefined ) { 
for ( name in object ) 
if ( callback.apply( object[ name ], args ) === false ) 
break; 
} else 
for ( ; i < length; ) 
if ( callback.apply( object[ i++ ], args ) === false ) 
break; 
} else { 
if ( length === undefined ) { 
for ( name in object ) 
if ( callback.call( object[ name ], name, object[ name ] ) === false ) 
break; 
} else 
for ( var value = object[0]; 
i < length && callback.call( value, i, value ) !== false; value = object[++i] ){} 
} 
return object; 
}

接着实现添加事件
jq里面是在 jQuery.event里面的add方法
在add方法里面实现了一下一些功能
取元素的events,handle这2个data绑定的数据
events存放的是事件列表
格式如下
{
click: [{handler:function(){},type:"click",guid:'xx'}.......],
mouse:[......]
}
handle是执行的函数
(所有的执行函数都是一样的 他们遍历事件列表 执行对应的事件)
然后遍历types 因为可以绑定多个事件
回调函数也会给几个属性
假设回调函数是handler
handler.guid = gevent.guid++
handler.type = name
name应该算一个特殊的命名 方便删除用的
比如
$('#xx')
.bind('click',function(){})
.bind('click.d',handler)
name就是d了
删除的时候可以只删除d那个事件 不删除上面的那个 click事件
最后是给元素绑定事件 但是执行的函数都是
function(){
gevent.handle.apply(arguments.callee.elem, arguments);
});
win.gevent = { 
guid : 1, 
add : function (elem, types, handler){ 
if ( elem.nodeType == 3 || elem.nodeType == 8 ) 
return; 
if ( elem.setInterval && elem != window ) 
elem = window; 
//给函数一个唯一标识的索引 方便后面删除该事件 
if ( !handler.guid ) 
handler.guid = this.guid++; 
//获得该元素的events handle 下的数据 
var events = data(elem, "events") || data(elem, "events", {}), 
handle =data(elem, "handle") || data(elem, "handle", function(){ 
//gevent.handle才是各种行为触发后会执行的函数 
gevent.handle.apply(arguments.callee.elem, arguments); 
}); 
handle.elem = elem; 
//遍历事件名 因为可以是 click mouseover 
each(types.split(/\s+/), function(index, type) { 
var namespaces = type.split("."); 
//获得事件名 
type = namespaces.shift(); 
//去掉点后面的东西 是个特殊的命名 在删除的时候可以指定删除他 如 click.d 
//用事件的type 记录住这个特殊的命名 
handler.type = namespaces.slice().sort().join("."); 
//获得该事件是否已经存在events 这个对象里面了 
var handlers = events[type]; 
//如果不存在该事件 给元素绑定该事件 
if (!handlers) { 
handlers = events[type] = {}; 
if (elem.addEventListener) 
elem.addEventListener(type, handle, false); 
else if (elem.attachEvent) 
elem.attachEvent("on" + type, handle); 
} 
//吧函数放到元素的该事件的列表里面 
handlers[handler.guid] = handler; 
}); 
elem = null; 
} 
}

gevent.hander是绑定事件真正执行的函数
在gevent.hander里面也有取.特殊命名的地方 但是不知道做什么用的
hander里面先对event进行包装
包装见gevent. fix 和 setEvent
主要是对做一个原生event的一个copy 然后把不兼容的方法 都合成兼容的写法
然后取元素的events (事件列表)
然后遍历这个事件列表 判断type是不是事件列表的key 是的话就执行事件
在执行列表函数的时候会判断返回值
如果返回false 还可以组织事件冒泡 和 默认行为
win.gevent = { 
handle : function(event){ 
var all, handlers; 
//包装event 
event = arguments[0] = gevent.fix( event || window.event ); event.currentTarget = this; 
//这里的........ 
var namespaces = event.type.split("."); 
event.type = namespaces.shift(); 
all = !namespaces.length; 
var namespace = RegExp("(^|\\.)" + namespaces.slice().sort().join(".*\\.") + "(\\.|$)"); 
//取这个元素的该行为 的 事件列表 
handlers = (data(this, "events") || {} )[event.type]; 
//遍历这个事件列表 执行该执行的东西 
for ( var j in handlers ) { 
var handler = handlers[j]; 
if ( all || namespace.test(handler.type) ) { 
// Pass in a reference to the handler function itself 
// So that we can later remove it 
// jq上的注释是是这么写的 把event的handler 引用这个事件 方便之后移除 
// 但是在remove里面 并没有用到event的handler 不知道这里到底有什么用 且有多个事件的时候这个事件被取代 
event.handler = handler; 
//执行事件 并且是用元素调用的事件 可以吧事件里面的this执行元素 ret为函数的返回值 
var ret = handler.apply(this, arguments); 
//如果有返回值 且返回值是false 执行阻止事件冒泡 阻止执行事件默认行为 
if( ret !== undefined ){ 
event.result = ret; 
if ( ret === false ) { 
event.preventDefault(); 
event.stopPropagation(); 
} 
} 
} 
} 
}, 
props: "altKey attrChange attrName bubbles button cancelable charCode clientX clientY ctrlKey currentTarget data detail eventPhase fromElement handler keyCode metaKey newValue originalTarget pageX pageY prevValue relatedNode relatedTarget screenX screenY shiftKey srcElement target toElement view wheelDelta which".split(" "), 
fix : function(event){ 
//new setEvent会给event给以个expando属性 如果有中个属性 说明已经生成了event了 不需要在次对event进行包装 
if ( event[expando] ) 
return event; 
//保留一个原始的event 
// new一个新的event 这个与原始的event是不同的 
var originalEvent = event; 
event = new setEvent( originalEvent ); 
//获得原始event的属性值 有哪些属性值 见 this.props 
for ( var i = this.props.length, prop; i; ){ 
prop = this.props[ --i ]; 
event[ prop ] = originalEvent[ prop ]; 
} 
//将目标元素同一成event.target 
if ( !event.target ) 
event.target = event.srcElement || document; // Fixes #1925 where srcElement might not be defined either 
//如果发现是文本节点 取他的父节点 
if ( event.target.nodeType == 3 ) 
event.target = event.target.parentNode; 
if ( !event.relatedTarget && event.fromElement ) 
event.relatedTarget = event.fromElement == event.target ? event.toElement : event.fromElement; 
return event; 
} 
} 
win.setEvent = function(src){ 
// Allow instantiation without the 'new' keyword 
// Event object 
if( src && src.type ){ 
this.originalEvent = src; 
this.type = src.type; 
// Event type 
}else 
this.type = src; 
// timeStamp is buggy for some events on Firefox(#3843) 
// So we won't rely on the native value 
this.timeStamp = now(); 
// Mark it as fixed 
this[expando] = true; 
} 
function returnFalse(){ 
return false; 
} 
function returnTrue(){ 
return true; 
} 
setEvent.prototype = { 
preventDefault: function() { 
var e = this.originalEvent; 
if( !e ) 
return; 
// if preventDefault exists run it on the original event 
if (e.preventDefault) 
e.preventDefault(); 
// otherwise set the returnValue property of the original event to false (IE) 
e.returnValue = false; 
}, 
stopPropagation: function() { 
var e = this.originalEvent; 
if( !e ) 
return; 
// if stopPropagation exists run it on the original event 
if (e.stopPropagation) 
e.stopPropagation(); 
// otherwise set the cancelBubble property of the original event to true (IE) 
e.cancelBubble = true; 
}, 
stopImmediatePropagation:function(){ 
this.isImmediatePropagationStopped = returnTrue; 
this.stopPropagation(); 
}, 
isImmediatePropagationStopped: returnFalse 
};

一个完整的例子
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 
<html xmlns="http://www.w3.org/1999/xhtml"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title></title> 
</head> 
<body> 
<div id="vv" style="height:200px;width:200px; background-color:#f00; padding:20px;"> 
<div id="xx" style="height:200px;width:200px; background-color:#000000;"></div> 
</div> 
<script type="text/javascript"> 
(function(doc,undefined){ 
function now(){ 
return +new Date; 
}; 
var win = this, 
expando = "jQuery" + now(), 
uuid = 0, 
cache = {}; 
win.data = function(elem, name, data){ 
var id = elem[expando]; 
if(!id) 
id = elem[expando] = ++uuid; 
if(name&&!cache[id]) 
cache[id] = {}; 
if(data !== undefined) 
cache[id][name] = data; 
return name 
? cache[id][name] 
: id; 
} 
win.removeData = function(elem, name){ 
var id = elem[expando]; 
if (name){ 
if (cache[id]) { 
delete cache[id][name]; 
name = ""; 
for ( name in cache[ id ] ) 
break; 
if ( !name ) 
removeData(elem); 
} 
}else{ 
try { 
delete elem[expando]; 
} catch(e){ 
if ( elem.removeAttribute ) 
elem.removeAttribute( expando ); 
} 
delete cache[id]; 
} 
} 
win.each = function( object, callback, args ) { 
var name, i = 0, length = object.length; 
if ( args ) { 
if ( length === undefined ) { 
for ( name in object ) 
if ( callback.apply( object[ name ], args ) === false ) 
break; 
} else 
for ( ; i < length; ) 
if ( callback.apply( object[ i++ ], args ) === false ) 
break; 
} else { 
if ( length === undefined ) { 
for ( name in object ) 
if ( callback.call( object[ name ], name, object[ name ] ) === false ) 
break; 
} else 
for ( var value = object[0]; 
i < length && callback.call( value, i, value ) !== false; value = object[++i] ){} 
} 
return object; 
} 
win.gevent = { 
guid : 1, 
add : function (elem, types, handler){ 
if ( elem.nodeType == 3 || elem.nodeType == 8 ) 
return; 
if ( elem.setInterval && elem != window ) 
elem = window; 
//给函数一个唯一标识的索引 方便后面删除该事件 
if ( !handler.guid ) 
handler.guid = this.guid++; 
//获得该元素的events handle 下的数据 
var events = data(elem, "events") || data(elem, "events", {}), 
handle =data(elem, "handle") || data(elem, "handle", function(){ 
//gevent.handle才是各种行为触发后会执行的函数 
gevent.handle.apply(arguments.callee.elem, arguments); 
}); 
handle.elem = elem; 
//遍历事件名 因为可以是 click mouseover 
each(types.split(/\s+/), function(index, type) { 
var namespaces = type.split("."); 
//获得事件名 
type = namespaces.shift(); 
//去掉点后面的东西 是个特殊的命名 在删除的时候可以指定删除他 如 click.d 
//用事件的type 记录住这个特殊的命名 
handler.type = namespaces.slice().sort().join("."); 
//获得该事件是否已经存在events 这个对象里面了 
var handlers = events[type]; 
//如果不存在该事件 给元素绑定该事件 
if (!handlers) { 
handlers = events[type] = {}; 
if (elem.addEventListener) 
elem.addEventListener(type, handle, false); 
else if (elem.attachEvent) 
elem.attachEvent("on" + type, handle); 
} 
//吧函数放到元素的该事件的列表里面 
handlers[handler.guid] = handler; 
}); 
elem = null; 
}, 
remove: function(elem, types, handler) { 
if ( elem.nodeType == 3 || elem.nodeType == 8 ) 
return; 
//获取这个元素的所有行为列表 如 {click:{},mouseocer:{}} 
var events = data(elem, "events"), ret, index; 
if(events){ 
//如果没出入行为类型 则删除这个元素的所有事件 
//如果传入的是.xx这种形式的 把所有行为的包含.xx命名的全部干掉 
if ( types === undefined || (typeof types === "string" && types.charAt(0) == ".") ){ 
for ( var type in events ) 
this.remove( elem, type + (types || "") ); 
}else{ 
//不知道干嘛的 
if ( types.type ) { 
handler = types.handler; 
types = types.type; 
} 
//因为删除事件可以一次支持删除多个 如click mouseover 所有要遍历删除 
each(types.split(/\s+/),function(index, type){ 
var namespaces = type.split("."); 
type = namespaces.shift(); 
var namespace = RegExp("(^|\\.)" + namespaces.slice().sort().join(".*\\.") + "(\\.|$)"); 
if ( events[type] ) { 
//如果传了第3个参数 函数 则删除这个事件 
if ( handler ) 
delete events[type][handler.guid]; 
else{ 
//遍历中个这个的所有行为 
for ( var handle in events[type] ){ 
// Handle the removal of namespaced events 
//删除有特殊命名的函数 
//如果没有特殊命名 正则 则是/^|..|$/ 可以匹配空 所以也能删除掉没有特殊命名的函数 
if ( namespace.test(events[type][handle].type) ) 
delete events[type][handle]; 
} 
} 
} 
for ( ret in events[type] ) break; 
//如果events[type]变成空的了 也就是{} 删除这个元素的的绑定事件 
if ( !ret ) { 
if (elem.removeEventListener) 
elem.removeEventListener(type, data(elem, "handle"), false); 
else if (elem.detachEvent) 
elem.detachEvent("on" + type, data(elem, "handle")); 
ret = null; 
delete events[type]; 
} 
}); 
} 
for ( ret in events ) break; 
//如果发现元素的整个events都是空的了 
//清空掉handle 并且清空掉他所有的引用 
if ( !ret ) { 
var handle = data( elem, "handle" ); 
if ( handle ) handle.elem = null; 
removeData( elem, "events" ); 
removeData( elem, "handle" ); 
} 
} 
}, 
handle : function(event){ 
var all, handlers; 
//包装event 
event = arguments[0] = gevent.fix( event || window.event ); 
event.currentTarget = this; 
//这里的........ 
var namespaces = event.type.split("."); 
event.type = namespaces.shift(); 
all = !namespaces.length; 
var namespace = RegExp("(^|\\.)" + namespaces.slice().sort().join(".*\\.") + "(\\.|$)"); 
//取这个元素的该行为 的 事件列表 
handlers = (data(this, "events") || {} )[event.type]; 
//遍历这个事件列表 执行该执行的东西 
for ( var j in handlers ) { 
var handler = handlers[j]; 
if ( all || namespace.test(handler.type) ) { 
// Pass in a reference to the handler function itself 
// So that we can later remove it 
// jq上的注释是是这么写的 把event的handler 引用这个事件 方便之后移除 
// 但是在remove里面 并没有用到event的handler 不知道这里到底有什么用 且有多个事件的时候这个事件被取代 
event.handler = handler; 
//执行事件 并且是用元素调用的事件 可以吧事件里面的this执行元素 ret为函数的返回值 
var ret = handler.apply(this, arguments); 
//如果有返回值 且返回值是false 执行阻止事件冒泡 阻止执行事件默认行为 
if( ret !== undefined ){ 
event.result = ret; 
if ( ret === false ) { 
event.preventDefault(); 
event.stopPropagation(); 
} 
} 
} 
} 
}, 
props: "altKey attrChange attrName bubbles button cancelable charCode clientX clientY ctrlKey currentTarget data detail eventPhase fromElement handler keyCode metaKey newValue originalTarget pageX pageY prevValue relatedNode relatedTarget screenX screenY shiftKey srcElement target toElement view wheelDelta which".split(" "), 
fix : function(event){ 
//new setEvent会给event给以个expando属性 如果有中个属性 说明已经生成了event了 不需要在次对event进行包装 
if ( event[expando] ) 
return event; 
//保留一个原始的event 
// new一个新的event 这个与原始的event是不同的 
var originalEvent = event; 
event = new setEvent( originalEvent ); 
//获得原始event的属性值 有哪些属性值 见 this.props 
for ( var i = this.props.length, prop; i; ){ 
prop = this.props[ --i ]; 
event[ prop ] = originalEvent[ prop ]; 
} 
//将目标元素同一成event.target 
if ( !event.target ) 
event.target = event.srcElement || document; // Fixes #1925 where srcElement might not be defined either 
//如果发现是文本节点 取他的父节点 
if ( event.target.nodeType == 3 ) 
event.target = event.target.parentNode; 
if ( !event.relatedTarget && event.fromElement ) 
event.relatedTarget = event.fromElement == event.target ? event.toElement : event.fromElement; 
return event; 
} 
} 
win.setEvent = function(src){ 
// Allow instantiation without the 'new' keyword 
// Event object 
if( src && src.type ){ 
this.originalEvent = src; 
this.type = src.type; 
// Event type 
}else 
this.type = src; 
// timeStamp is buggy for some events on Firefox(#3843) 
// So we won't rely on the native value 
this.timeStamp = now(); 
// Mark it as fixed 
this[expando] = true; 
} 
function returnFalse(){ 
return false; 
} 
function returnTrue(){ 
return true; 
} 
setEvent.prototype = { 
preventDefault: function() { 
var e = this.originalEvent; 
if( !e ) 
return; 
// if preventDefault exists run it on the original event 
if (e.preventDefault) 
e.preventDefault(); 
// otherwise set the returnValue property of the original event to false (IE) 
e.returnValue = false; 
}, 
stopPropagation: function() { 
var e = this.originalEvent; 
if( !e ) 
return; 
// if stopPropagation exists run it on the original event 
if (e.stopPropagation) 
e.stopPropagation(); 
// otherwise set the cancelBubble property of the original event to true (IE) 
e.cancelBubble = true; 
}, 
stopImmediatePropagation:function(){ 
this.isImmediatePropagationStopped = returnTrue; 
this.stopPropagation(); 
}, 
isImmediatePropagationStopped: returnFalse 
}; 
})(document); 
var $ = function(id){return document.getElementById(id)} 
var a = function(){alert(1)} 
window.onload = function(){ 
gevent.add($('xx'),'click',a); 
gevent.add($('xx'),'click',function(){alert(1)}); 
gevent.add($('xx'),'click',function(){alert(2)}); 
gevent.add($('xx'),'click',function(){alert(3)}); 
gevent.add($('xx'),'click.xx',function(){alert(4)}); 
} 
</script> 
</body> 
</html>
Javascript 相关文章推荐
使用ExtJS技术实现的拖动树结点
Aug 05 Javascript
使用js判断TextBox控件值改变然后出发事件
Mar 07 Javascript
Javascript 正则表达式实现为数字添加千位分隔符
Mar 10 Javascript
jQuery实现仿新浪微博浮动的消息提示框(可智能定位)
Oct 10 Javascript
Node.js DES加密的简单实现
Jul 07 Javascript
BootStrap的table表头固定tbody滚动的实例代码
Aug 24 Javascript
使用openSpeDiv方法实现Ecshop登录弹窗框效果
Mar 13 Javascript
基于node.js的fs核心模块读写文件操作(实例讲解)
Sep 10 Javascript
JavaScript实现快速排序的方法分析
Jan 10 Javascript
使用jQuery实现掷骰子游戏
Oct 24 jQuery
JavaScript类的继承多种实现方法
May 30 Javascript
JS如何判断对象是否包含某个属性
Aug 29 Javascript
JQuery扩展插件Validate—6 radio、checkbox、select的验证
Sep 05 #Javascript
JQuery扩展插件Validate—4设置错误提示的样式
Sep 05 #Javascript
JQuery扩展插件Validate 3通过参数设置错误信息
Sep 05 #Javascript
JQuery扩展插件Validate 2通过参数设置验证规则
Sep 05 #Javascript
JQuery扩展插件Validate 1 基本使用方法并打包下载
Sep 05 #Javascript
JQuery扩展插件Validate 5添加自定义验证方法
Sep 05 #Javascript
分享20多个很棒的jQuery 文件上传插件或教程
Sep 04 #Javascript
You might like
实用函数4
2007/11/08 PHP
在PHP中养成7个面向对象的好习惯
2010/07/17 PHP
基于python发送邮件的乱码问题的解决办法
2013/04/25 PHP
php-cli简介(不会Shell语言一样用Shell)
2013/06/03 PHP
在win系统安装配置 Memcached for PHP 5.3 图文教程
2015/03/03 PHP
PHP实现生成带背景的图形验证码功能
2016/10/03 PHP
Mootools 1.2教程 输入过滤第二部分(字符串)
2009/09/15 Javascript
jquery 弹出层注册页面等(asp.net后台)
2010/06/17 Javascript
JS控件ASP.NET的treeview控件全选或者取消(示例代码)
2013/12/16 Javascript
js控制分页打印、打印分页示例
2014/02/08 Javascript
Node.js中AES加密和其它语言不一致问题解决办法
2014/03/10 Javascript
js实现商品抛物线加入购物车特效
2020/11/18 Javascript
基于Bootstrap仿淘宝分页控件实现代码
2016/11/07 Javascript
js处理层级数据结构的方法小结
2017/01/17 Javascript
基于jQuery实现瀑布流页面
2017/04/11 jQuery
微信小程序实战之自定义toast(6)
2017/04/18 Javascript
解决node修改后需频繁手动重启的问题
2018/05/13 Javascript
vue中动态设置meta标签和title标签的方法
2018/07/11 Javascript
浅谈laytpl 模板空值显示null的解决方法及简单的js表达式
2019/09/19 Javascript
基于python爬虫数据处理(详解)
2017/06/10 Python
Python编程实现两个文件夹里文件的对比功能示例【包含内容的对比】
2017/06/20 Python
python爬取个性签名的方法
2018/06/17 Python
Python 生成 -1~1 之间的随机数矩阵方法
2018/08/04 Python
在Pycharm中项目解释器与环境变量的设置方法
2018/10/29 Python
python 字符串只保留汉字的方法
2018/11/16 Python
Django自定义用户登录认证示例代码
2019/06/30 Python
解决win7操作系统Python3.7.1安装后启动提示缺少.dll文件问题
2019/07/15 Python
pycharm 添加解释器的方法步骤
2020/08/31 Python
python中函数返回多个结果的实例方法
2020/12/16 Python
软件配置管理有什么好处
2015/04/15 面试题
自我鉴定书面格式
2014/01/13 职场文书
结对共建工作方案
2014/06/02 职场文书
庆七一活动总结
2014/08/27 职场文书
授权委托书(公民个人适用)
2014/09/19 职场文书
python 算法题——快乐数的多种解法
2021/05/27 Python
Python开发五子棋小游戏
2022/05/02 Python