博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
jQuery代码优化:基本事件
阅读量:7099 次
发布时间:2019-06-28

本文共 2226 字,大约阅读时间需要 7 分钟。

jQuery对事件系统的抽象与优化也是它的一大特色。本文仅从事件系统入手,简要分析一下jQuery为什么提供mouseenter和mouseleave事件,它们与标准的mouseover、mouseout事件有什么区别。

事件模型

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

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

 

1
2
3
4
5
<
div
>
    
<
span
>
        
<
a
>...</
a
>
    
</
span
>
</
div
>

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

 

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

 

1
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提供支持的。翻译了一下注释,仅供大家参考。

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
// 下面这个函数用于检测事件是否发生在另一个元素的内部
// 在 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事件来避免事件冒泡的副作用。

转载于:https://www.cnblogs.com/ranran/p/3631091.html

你可能感兴趣的文章
报警系统QuickAlarm之报警执行器的设计与实现
查看>>
记一次docker部署SpringBoot项目乱码
查看>>
记录一次基于vue、typescript、pwa的项目由开发到部署
查看>>
Vue——关于自定义组件
查看>>
Koa引用库之Koa-compose
查看>>
手机抓包+注入黑科技HttpCanary——最强大的Android抓包注入工具
查看>>
JavaScript读书笔记(对象的总结)
查看>>
【数据挖掘】用文本挖掘和机器学习洞悉数据
查看>>
深入解析DC/OS 1.8 – 高可靠的微服务及大数据管理平台
查看>>
LLVM & Clang 入门
查看>>
java基础知识精华总结(一)
查看>>
Zookeeper集群安装
查看>>
使用beego框架开发个人博客(一)
查看>>
《JavaScript语言精粹》学习笔记一
查看>>
js:字符串的一些操作方法?
查看>>
从零开始实现一个MQTT客户端 开篇漫谈
查看>>
谈一谈Flutter中的共享元素动画Hero
查看>>
vue在低版本的iOS下出现白屏现象解决方案
查看>>
vim从入门到弃坑:基础指令的归类
查看>>
[译] 苹果公司如何修复 3D Touch
查看>>