JavaScript事件委托的技术原理探讨示例


Posted in Javascript onApril 17, 2014

如今的JavaScript技术界里最火热的一项技术应该是‘事件委托(event delegation)'了。使用事件委托技术能让你避免对特定的每个节点添加事件监听器;相反,事件监听器是被添加到它们的父元素上。事件监听器会分析从子元素冒泡上来的事件,找到是哪个子元素的事件。基本概念非常简单,但仍有很多人不理解事件委托的工作原理。这里我将要解释事件委托是如何工作的,并提供几个纯JavaScript的基本事件委托的例子。

假定我们有一个UL元素,它有几个子元素:

<ul id="parent-list"> 
<li id="post-1">Item 1</li> 
<li id="post-2">Item 2</li> 
<li id="post-3">Item 3</li> 
<li id="post-4">Item 4</li> 
<li id="post-5">Item 5</li> 
<li id="post-6">Item 6</li> 
</ul>

我们还假设,当每个子元素被点击时,将会有各自不同的事件发生。你可以给每个独立的li元素添加事件监听器,但有时这些li元素可能会被删除,可能会有新增,监听它们的新增或删除事件将会是一场噩梦,尤其是当你的监听事件的代码放在应用的另一个地方时。但是,如果你将监听器安放到它们的父元素上呢?你如何能知道是那个子元素被点击了?

简单:当子元素的事件冒泡到父ul元素时,你可以检查事件对象的target属性,捕获真正被点击的节点元素的引用。下面是一段很简单的JavaScript代码,演示了事件委托的过程:

// 找到父元素,添加监听器... 
document.getElementById("parent-list").addEventListener("click",function(e) { 
// e.target是被点击的元素! 
// 如果被点击的是li元素 
if(e.target && e.target.nodeName == "LI") { 
// 找到目标,输出ID! 
console.log("List item ",e.target.id.replace("post-")," was clicked!"); 
} 
});

第一步是给父元素添加事件监听器。当有事件触发监听器时,检查事件的来源,排除非li子元素事件。如果是一个li元素,我们就找到了目标!如果不是一个li元素,事件将被忽略。这个例子非常简单,UL和li是标准的父子搭配。让我们试验一些差异比较大的元素搭配。假设我们有一个父元素div,里面有很多子元素,但我们关心的是里面的一个带有”classA” CSS类的A标记:
// 获得父元素DIV, 添加监听器... 
document.getElementById("myDiv").addEventListener("click",function(e) { 
// e.target是被点击的元素 
if(e.target && e.target.nodeName == "A") { 
// 获得CSS类名 
var classes = e.target.className.split(" "); 
// 搜索匹配! 
if(classes) { 
// For every CSS class the element has... 
for(var x = 0; x < classes.length; x++) { 
// If it has the CSS class we want... 
if(classes[x] == "classA") { 
// Bingo! 
console.log("Anchor element clicked!"); 
// Now do something here.... 
} 
} 
} 
} 
});

上面这个例子中不仅比较了标签名,而且比较了CSS类名。虽然稍微复杂了一点,但还是很具代表性的。比如,如果某个A标记里有一个span标记,则这个span将会成为target元素。这个时候,我们需要上溯DOM树结构,找到里面是否有一个 A.classA 的元素。

因为大部分程序员都会使用jQuery等工具库来处理DOM元素和事件,我建议大家都使用里面的事件委托方法,因为这里工具库里都提供了高级的委托方法和元素甄别方法。

希望这篇文章能帮助你理解JavaScript事件委托的幕后原理,希望你也感受到了事件委托的强大用处!

Javascript 相关文章推荐
javascript parseInt与Number函数的区别
Jan 21 Javascript
jquery getScript动态加载JS方法改进详解
Nov 15 Javascript
jquery获取特定name所有选中的checkbox,支持IE9标准模式
Mar 18 Javascript
jquery幻灯片插件bxslider样式改进实例
Oct 15 Javascript
利用jQuery插件imgAreaSelect实现获得选择域的图像信息
Dec 02 Javascript
如何给ss bash 写一个 WEB 端查看流量的页面
Mar 23 Javascript
微信小程序 自定义Toast实例代码
Jun 12 Javascript
微信小程序 获取二维码实例详解
Jun 23 Javascript
jQuery实现html table行Tr的复制、删除、计算功能
Jul 10 jQuery
vue教程之toast弹框全局调用示例详解
Aug 24 Javascript
基于Vue的商品主图放大镜方案详解
Sep 19 Javascript
用webAPI实现图片放大镜效果
Nov 23 Javascript
JS实现div居中示例
Apr 17 #Javascript
淘宝网提供的国内NPM镜像简介和使用方法
Apr 17 #Javascript
js调用后台、后台调用前台等方法总结
Apr 17 #Javascript
JS下载文件|无刷新下载文件示例代码
Apr 17 #Javascript
你可能不知道的JavaScript的new Function()方法
Apr 17 #Javascript
在JS中解析HTML字符串示例代码
Apr 16 #Javascript
iframe的onreadystatechange事件在firefox下的使用
Apr 16 #Javascript
You might like
超小PHP小马小结(方便查找后门的朋友)
2012/05/05 PHP
jQuery Mobile + PHP实现文件上传
2014/12/12 PHP
代码生成器 document.write()
2007/04/15 Javascript
兼容FireFox 的 js 日历 支持时间的获取
2009/03/04 Javascript
javascript new一个对象的实质
2010/01/07 Javascript
jquery 简单的进度条实现代码
2010/03/11 Javascript
jQuery获取注册信息并提示实现代码
2013/04/21 Javascript
js将字符串转成正则表达式的实现方法
2013/11/13 Javascript
ECMAScript6的新特性箭头函数(Arrow Function)详细介绍
2014/06/07 Javascript
jquery选择器简述
2015/08/31 Javascript
jQuery+AJAX实现遮罩层登录验证界面(附源码)
2020/09/13 Javascript
JavaScript常用数组算法小结
2016/02/13 Javascript
JavaScript常用函数工具集:lao-utils
2016/03/01 Javascript
ES6新特征数字、数组、字符串
2016/10/01 Javascript
JavaScript动态检验密码强度的实现方法
2016/11/09 Javascript
Node.js获取前端ajax提交的request信息
2017/02/20 Javascript
了解VUE的render函数的使用
2017/06/08 Javascript
JavaScript多态与封装实例分析
2018/07/27 Javascript
怎样使你的 JavaScript 代码简单易读(推荐)
2019/04/16 Javascript
JavaScript Math对象和调试程序的方法分析
2019/05/13 Javascript
JavaScript遍历查找数组中最大值与最小值的方法示例
2019/05/24 Javascript
python虚拟环境 virtualenv的简单使用
2020/01/21 Javascript
Python爬虫之pandas基本安装与使用方法示例
2018/08/08 Python
Python3.x+pyqtgraph实现数据可视化教程
2020/03/14 Python
Django {{ MEDIA_URL }}无法显示图片的解决方式
2020/04/07 Python
Python3爬虫中Ajax的用法
2020/07/10 Python
CSS3制作精致的照片墙特效
2016/06/07 HTML / CSS
html5 http的轮询和Websocket原理
2018/10/19 HTML / CSS
AmazeUI 按钮交互的实现示例
2020/08/24 HTML / CSS
Guess美国官网:美国知名服装品牌
2019/04/08 全球购物
爱岗敬业演讲稿
2014/05/05 职场文书
四风问题班子对照检查材料
2014/09/27 职场文书
小学家长通知书评语
2014/12/31 职场文书
如何写辞职信
2015/05/13 职场文书
创业计划书之少年玩具店
2019/09/05 职场文书
部分武汉产收音机展览
2022/04/07 无线电