jQuery代码优化之基本事件


Posted in Javascript onNovember 01, 2011

事件模型

说到事件,就要追溯到网景与微软的“浏览器大战”了。当时,事件模型还没有标准,两家公司的实现就是事实标准。网景在Navigator中实现了“事件捕获”的事件系统,而微软则在IE中实现了一个基本上相反的事件系统,叫做“事件冒泡”。这两种系统的区别在于当事件发生时,相关元素处理(响应)事件的优先权不同。

下面举例说明这两种事件机制的区别。假设文档中有如下结构:

<div> 
<span> 
<a>...</a> 
</span> 
</div>

因为这三个元素是嵌套的,所以单击了a,实际上也就单击了span和div。换句话说,这三个元素都应该有处理单击事件的机会。在事件捕获机制下,处理这个单击事件的优先次序是:div > span > a;而在事件冒泡机制下,处理这个单击事件的优先次序则是:a > span > div。

后来,W3C的规范要求浏览器同时支持捕获和冒泡机制,并允许开发人员选择把事件注册到哪个阶段。于是就有了下面这个注册事件的标准方法:

target.addEventListener(type, listener, useCapture Optional );

其中:

◆ type:字符串,表示监听的事件类型

◆ listener:监听器对象(JavaScript函数),在指定事件发生时可以收到通知

◆ useCapture:布尔值,是否注册到捕获阶段

在实际应用开发中,为了确保与IE(因为它不支持捕获)兼容,useCapture一般都指定为false(默认值也是false)。换句话说,只把事件注册到冒泡阶段;对于上面那个简单的例子来说,响应顺序就是:a > span > div。

冒泡的副作用

如前所述,IE的冒泡事件模型基本上成为了事实标准。但冒泡有一个副作用。

仍以前面的文档结构为例,假设它是界面中的一个菜单项,我们希望用户鼠标离开div时隐藏菜单。于是,我们给div注册了一个mouseout事件。如果用户鼠标是从div离开的,那么一切正确。而如果用户鼠标是从a或span离开的,问题就来了。因为由于事件冒泡,从这两个元素开始分派的mouseout事件都会传播到div,从而导致鼠标并没有离开div,菜单就提前隐藏了。

当然,冒泡的副作用不难避免。比如,给div内部的每个元素都注册mouseout事件,并使用.stopPropagation()方法阻止事件进一步传播。对于IE,就得将事件对象的cancelBubble属性设置为false,取消事件冒泡。不过,这仍然回到自己处理浏览器不兼容性问题的老路上了。

优化方案

为了避免冒泡的副作用,jQuery提供了mouseenter和mouseleave事件,就使用它们来代替mouseover和mouseout吧。

下面这个摘自jQuery的内部函数withinElement,就是为mouseenter和mouseleave提供支持的。翻译了一下注释,仅供大家参考。

// 下面这个函数用于检测事件是否发生在另一个元素的内部 
// 在 jQuery.event.special.mouseenter 和 mouseleave 处理程序中使用 
var withinElement = function( event ) { 
// 检测 mouse(over|out) 是否还在相同的父元素内 
var parent = event.relatedTarget; 
// 设置正确的事件类型 
event.type = event.data; 
// Firefox 有时候会把 relatedTarget 指定一个 XUL 元素 
// 对于这种元素,无法访问其 parentNode 属性 
try { 
// Chrome 也类似,虽然可以访问 parentNode 属性 
// 但结果却是 null 
if ( parent && parent !== document && !parent.parentNode ) { 
return; 
} 
// 沿 DOM 树向上 
while ( parent && parent !== this ) { 
parent = parent.parentNode; 
} 
if ( parent !== this ) { 
// 如果实际正好位于一个非子元素上面,那好,就处理事件 
jQuery.event.handle.apply( this, arguments ); 
} 
// 假定已经离开了元素,因为很可能鼠标放在了一个XUL元素上 
} catch(e) { } 
},

结论

在jQuery里,可以使用mouseenter和mouseleave事件来避免事件冒泡的副作用。
原文:http://www.ituring.com.cn/article/420

Javascript 相关文章推荐
分享精心挑选的12款优秀jQuery Ajax分页插件和教程
Aug 09 Javascript
web开发人员学习jQuery的6大理由及jQuery的优势介绍
Jan 03 Javascript
在父页面调用子页面的JS方法
Sep 29 Javascript
得到form下的所有的input的js代码
Nov 07 Javascript
JS实现div居中示例
Apr 17 Javascript
JavaScript indexOf方法入门实例(计算指定字符在字符串中首次出现的位置)
Oct 17 Javascript
Bootstrap4一次重大更新 几乎涉及每行代码
May 16 Javascript
VSCode配置react开发环境的步骤
Dec 27 Javascript
使用webpack打包koa2 框架app
Feb 02 Javascript
vue 实现在函数中触发路由跳转的示例
Sep 01 Javascript
Ant-design-vue Table组件customRow属性的使用说明
Oct 28 Javascript
JS封装cavans多种滤镜组件
Feb 15 Javascript
js下获得客户端操作系统的函数代码(1:vista,2:windows7,3:2000,4:xp,5:2003,6:2008)
Oct 31 #Javascript
线路分流自动智能跳转代码,自动选择最快镜像网站(js)
Oct 31 #Javascript
IE与Firefox在JavaScript上的7个不同句法分享
Oct 30 #Javascript
加载 Javascript 最佳实践
Oct 30 #Javascript
js判断是否为数组的函数: isArray()
Oct 30 #Javascript
JS trim去空格的最佳实践
Oct 30 #Javascript
js中更短的 Array 类型转换
Oct 30 #Javascript
You might like
phpMyadmin 用户权限中英对照
2010/04/02 PHP
php 无法加载mcrypt.dll的解决办法
2013/04/03 PHP
PHP获取浏览器信息类和客户端地理位置的2个方法
2014/04/24 PHP
PHP实现将浏览历史页面网址保存到cookie的方法
2015/01/26 PHP
PHP可变变量学习小结
2015/11/29 PHP
PHP中addslashes与mysql_escape_string的区别分析
2016/04/25 PHP
深入剖析浏览器退出之后php还会继续执行么
2016/05/17 PHP
php使用json-schema模块实现json校验示例
2019/09/28 PHP
在JavaScript中实现命名空间
2006/11/23 Javascript
javascript 关闭IE6、IE7
2009/06/01 Javascript
js/jquery解析json和数组格式的方法详解
2014/01/09 Javascript
JavaScript中的值是按值传递还是按引用传递问题探讨
2015/01/30 Javascript
浅谈jQuery animate easing的具体使用方法(推荐)
2016/06/17 Javascript
vue.js指令v-model使用方法
2017/03/20 Javascript
Bootstrap栅格系统的使用详解
2017/10/30 Javascript
vue.js项目中实用的小技巧汇总
2017/11/29 Javascript
详解Vue中数组和对象更改后视图不刷新的问题
2018/09/21 Javascript
通过扫小程序码实现网站登陆功能
2019/08/22 Javascript
layui输入框只允许输入中文且判断长度的例子
2019/09/18 Javascript
使用layui的layer组件做弹出层的例子
2019/09/27 Javascript
[06:11]2014DOTA2国际邀请赛 专访团结一心的VG战队
2014/07/21 DOTA
[52:07]完美世界DOTA2联赛PWL S3 LBZS vs access 第二场 12.10
2020/12/13 DOTA
Python实现读取TXT文件数据并存进内置数据库SQLite3的方法
2017/08/08 Python
使用 Python 实现微信公众号粉丝迁移流程
2018/01/03 Python
python kmeans聚类简单介绍和实现代码
2018/02/23 Python
Python统计一个字符串中每个字符出现了多少次的方法【字符串转换为列表再统计】
2019/05/05 Python
Python读取xlsx文件的实现方法
2019/07/04 Python
video下autoplay属性无效的解决方法(添加muted属性)
2020/05/19 HTML / CSS
听课评语大全
2014/04/30 职场文书
村容村貌整治方案
2014/05/21 职场文书
教师节宣传方案
2014/05/23 职场文书
2014领导班子四风问题对照检查材料思想汇报
2014/09/21 职场文书
家庭贫困证明范本(经典版)
2014/09/22 职场文书
自主招生推荐信怎么写
2015/03/26 职场文书
新闻通讯稿范文
2015/07/22 职场文书
python异常中else的实例用法
2021/06/15 Python