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 相关文章推荐
纯JavaScript实现的兼容各浏览器的添加和移除事件封装
Mar 28 Javascript
JavaScript使用shift方法移除素组第一个元素实例分析
Apr 06 Javascript
jQuery基于ajax实现星星评论代码
Aug 07 Javascript
快速解决jquery.touchSwipe左右滑动和垂直滚动条冲突
Apr 15 Javascript
实例讲解Jquery中隐藏hide、显示show、切换toggle的用法
May 13 Javascript
js事件驱动机制 浏览器兼容处理方法
Jul 23 Javascript
javascript cookie基础应用之记录用户名的方法
Sep 20 Javascript
Vue.js实战之通过监听滚动事件实现动态锚点
Apr 04 Javascript
js 开发之autocomplete=&quot;off&quot;在chrom中失效的解决办法
Sep 28 Javascript
JavaScript自执行函数和jQuery扩展方法详解
Oct 27 jQuery
基于JSONP原理解析(推荐)
Dec 04 Javascript
关于JavaScript轮播图的实现
Nov 20 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
使用PHP编写的SVN类
2013/07/18 PHP
简单的php文件上传(实例)
2013/10/27 PHP
PHP利用MySQL保存session的实现思路及示例代码
2014/09/09 PHP
分享一则PHP定义函数代码
2015/02/26 PHP
javascript中encodeURI和decodeURI方法使用介绍
2013/05/06 Javascript
jQuery对html元素取值与赋值的方法
2013/11/20 Javascript
深入理解JavaScript系列(30):设计模式之外观模式详解
2015/03/03 Javascript
使用JSON作为函数的参数的优缺点
2016/10/27 Javascript
微信小程序 页面传参实例详解
2016/11/16 Javascript
20行js代码实现的贪吃蛇小游戏
2017/06/20 Javascript
关于react-router的几种配置方式详解
2017/07/24 Javascript
原生JS实现的放大镜特效示例【测试可用】
2018/12/08 Javascript
vue学习笔记之给组件绑定原生事件操作示例
2020/02/27 Javascript
js实现简易点击切换显示或隐藏
2020/11/29 Javascript
vue3.0+vue-router+element-plus初实践
2020/12/02 Vue.js
vue 计算属性和侦听器的使用小结
2021/01/25 Vue.js
[00:10]DOTA2全国高校联赛 以DOTA2会友
2018/05/30 DOTA
Python实现的一个自动售饮料程序代码分享
2014/08/25 Python
基于Django框架的权限组件rbac实例讲解
2019/08/31 Python
python命令 -u参数用法解析
2019/10/24 Python
python实现跨excel sheet复制代码实例
2020/03/03 Python
Python3 搭建Qt5 环境的方法示例
2020/07/16 Python
Css3+Js制作漂亮时钟(附源码)
2013/04/24 HTML / CSS
HTML5的Geolocation地理位置定位API使用教程
2016/05/12 HTML / CSS
美国精品家居用品网站:US-Mattress
2016/08/24 全球购物
植物选择:Botanic Choice
2017/02/15 全球购物
公务员诚信承诺书
2014/05/26 职场文书
真诚的求职信
2014/07/04 职场文书
预备党员半年考察意见
2015/06/01 职场文书
公司车队管理制度
2015/08/04 职场文书
Vue图片裁剪组件实例代码
2021/07/02 Vue.js
使用canvas对video视频某一刻截图功能
2021/09/25 HTML / CSS
简单聊一聊SQL注入及防止SQL注入
2022/03/23 MySQL
AJAX引擎原理以及XmlHttpRequest对象的axios、fetch区别详解
2022/04/09 Javascript
python游戏开发之pygame实现接球小游戏
2022/04/22 Python
本地搭建minio文件服务器(使用bat脚本启动)的方法
2022/07/15 Servers