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 相关文章推荐
jQuery 操作下拉列表框实现代码
Feb 22 Javascript
Prototype的Class.create函数解析
Sep 22 Javascript
提取字符串中年月日的函数代码
Nov 05 Javascript
两种方法实现在HTML页面加载完毕后运行某个js
Jun 16 Javascript
使用基于Node.js的构建工具Grunt来发布ASP.NET MVC项目
Feb 15 Javascript
浅谈js中的in-for循环
Jun 28 Javascript
JS实现数字格式千分位相互转换方法
Aug 01 Javascript
AngularJS入门教程之路由机制ngRoute实例分析
Dec 13 Javascript
Vue中的v-cloak使用解读
Mar 27 Javascript
vue项目添加多页面配置的步骤详解
May 22 Javascript
转换layUI的数据表格中的日期格式方法
Sep 19 Javascript
vue总线机制(bus)知识点详解
May 10 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
学习discuz php 引入文件的方法DISCUZ_ROOT
2009/06/21 PHP
PHP使用递归生成文章树
2015/04/21 PHP
PHP使用递归方式列出当前目录下所有文件的方法
2015/06/02 PHP
php模拟post上传图片实现代码
2016/06/24 PHP
利用javascript解决图片缩放及其优化的代码
2012/05/23 Javascript
jQuery 回车事件enter使用示例
2014/02/18 Javascript
js function定义函数的几种不错方法
2014/02/27 Javascript
JS实现简单的顶部定时关闭层效果
2014/06/15 Javascript
最精简的JavaScript实现鼠标拖动效果的方法
2015/05/11 Javascript
Vue2 SSR渲染根据不同页面修改 meta
2017/11/20 Javascript
Angular模版驱动表单的使用总结
2018/05/05 Javascript
NodeJS 实现多语言的示例代码
2018/09/11 NodeJs
详解三种方式解决vue中v-html元素中标签样式
2018/11/22 Javascript
微信小程序getLocation 需要在app.json中声明permission字段
2020/03/03 Javascript
用python删除java文件头上版权信息的方法
2014/07/31 Python
在Python中使用dict和set方法的教程
2015/04/27 Python
python下载图片实现方法(超简单)
2017/07/21 Python
Python和Java进行DES加密和解密的实例
2018/01/09 Python
python版本单链表实现代码
2018/09/28 Python
Python2和3字符编码的区别知识点整理
2019/08/08 Python
Python面向对象之多态原理与用法案例分析
2019/12/30 Python
Anaconda+vscode+pytorch环境搭建过程详解
2020/05/25 Python
Python使用shutil模块实现文件拷贝
2020/07/31 Python
纯CSS改变webkit内核浏览器的滚动条样式
2014/04/17 HTML / CSS
利用css3实现的简单的鼠标悬停按钮
2014/11/04 HTML / CSS
html5图片上传预览示例分享
2014/04/14 HTML / CSS
全球最大的在线旅游公司:Expedia
2017/11/16 全球购物
The Hut英国:英国领先的豪华在线百货商店
2019/07/26 全球购物
大学生水果店创业计划书
2014/01/28 职场文书
学校大课间活动方案
2014/01/30 职场文书
经销商年会策划方案
2014/05/29 职场文书
被告代理词范文
2015/05/25 职场文书
2015双创工作总结
2015/07/24 职场文书
mysql 索引的数据结构为什么要采用B+树
2022/04/26 MySQL
js前端面试常见浏览器缓存强缓存及协商缓存实例
2022/06/21 Javascript
Win11 21h2可以升级22h2吗?看看你的电脑符不符合要求
2022/07/07 数码科技