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中字符串拼接详解
Sep 26 Javascript
有效提高JavaScript执行效率的几点知识
Jan 31 Javascript
JavaScript获取表单内所有元素值的方法
Apr 02 Javascript
jquery关于事件冒泡和事件委托的技巧及阻止与允许事件冒泡的三种实现方法
Nov 27 Javascript
Vuejs第十篇之vuejs父子组件通信
Sep 06 Javascript
JS实现页面载入时随机显示图片效果
Sep 07 Javascript
ionic cordova一次上传多张图片(类似input file提交表单)的实现方法
Dec 16 Javascript
Vue2.0 多 Tab切换组件的封装实例
Jul 28 Javascript
详解小程序输入框闪烁及重影BUG解决方案
Aug 31 Javascript
uniapp与webview之间的相互传值的实现
Jun 29 Javascript
jquery实现简单拖拽效果
Jul 20 jQuery
Nuxt 项目性能优化调研分析
Nov 07 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
上海永华YH-R296(华普R-96)12波段立体声收音机的分析和打理
2021/03/02 无线电
适用于php-5.2 的 php.ini 中文版[金步国翻译]
2011/04/17 PHP
PHP字符串的编码问题的详细介绍
2013/04/27 PHP
yii2简单使用less代替css示例
2017/03/10 PHP
nodejs获取本机内网和外网ip地址的实现代码
2014/06/01 NodeJs
javascript实现通过表格绘制颜色填充矩形的方法
2015/04/21 Javascript
webpack-url-loader 解决项目中图片打包路径问题
2019/02/15 Javascript
基于Vue2-Calendar改进的日历组件(含中文使用说明)
2019/04/14 Javascript
node Buffer缓存区常见操作示例
2019/05/04 Javascript
javaScript把其它类型转换为Number类型
2019/10/13 Javascript
React学习之JSX与react事件实例分析
2020/01/06 Javascript
js实现浏览器打印功能的示例代码
2020/07/15 Javascript
[03:35]2018年度DOTA2最佳辅助位选手5号位-完美盛典
2018/12/17 DOTA
[37:45]完美世界DOTA2联赛PWL S3 LBZS vs Phoenix 第二场 12.09
2020/12/11 DOTA
python中readline判断文件读取结束的方法
2014/11/08 Python
Python通过select实现异步IO的方法
2015/06/04 Python
浅谈编码,解码,乱码的问题
2016/12/30 Python
python利用dir函数查看类中所有成员函数示例代码
2017/09/08 Python
python中如何使用正则表达式的非贪婪模式示例
2017/10/09 Python
Tensorflow 同时载入多个模型的实例讲解
2018/07/27 Python
浅述python2与python3的简单区别
2018/09/19 Python
python opencv 读取本地视频文件 修改ffmpeg的方法
2019/01/26 Python
python 通过SSHTunnelForwarder隧道连接redis的方法
2019/02/19 Python
python字典setdefault方法和get方法使用实例
2019/12/25 Python
pytorch如何冻结某层参数的实现
2020/01/10 Python
Django ForeignKey与数据库的FOREIGN KEY约束详解
2020/05/20 Python
使用python创建Excel工作簿及工作表过程图解
2020/05/27 Python
通用的Django注册功能模块实现方法
2021/02/05 Python
Aveda美国官网:天然护发产品、洗发水、护发素和沙龙
2016/12/09 全球购物
财务总监岗位职责范本
2015/04/03 职场文书
建国大业电影观后感
2015/06/01 职场文书
新课程改革心得体会
2016/01/22 职场文书
什么是执行力?9个故事告诉您:成功绝非偶然!
2019/07/05 职场文书
简单实现一个手持弹幕功能+文字抖动特效
2021/03/31 HTML / CSS
详解如何使用Nginx解决跨域问题
2022/05/06 Servers
Android Studio实现带三角函数对数运算功能的高级计算器
2022/05/20 Java/Android