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 相关文章推荐
用jquery等比例控制图片宽高的具体实现
Jan 28 Javascript
Iframe实现跨浏览器自适应高度解决方法
Sep 02 Javascript
JavaScript中停止执行setInterval和setTimeout事件的方法
May 14 Javascript
快速掌握WordPress中加载JavaScript脚本的方法
Dec 17 Javascript
js只执行1次的函数示例
Jul 20 Javascript
JS实现图片高斯模糊切换效果的焦点图实例
Jan 21 Javascript
ReactNative中使用Redux架构总结
Dec 15 Javascript
详解基于electron制作一个node压缩图片的桌面应用
Jan 29 Javascript
微信小程序用户拒绝授权的处理方法详解
Sep 20 Javascript
JS造成内存泄漏的几种情况实例分析
Mar 02 Javascript
微信小程序间使用navigator跳转传值问题实例分析
Mar 27 Javascript
jQuery实现tab栏切换效果
Dec 22 jQuery
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
利用ThinkPHP内置的ThinkAjax实现异步传输技术的实现方法
2011/12/19 PHP
简单的php中文转拼音的实现代码
2014/02/11 PHP
PHP strip_tags()去除HTML、XML以及PHP的标签介绍
2014/02/18 PHP
通过修改配置真正解决php文件上传大小限制问题(nginx+php)
2015/09/23 PHP
thinkphp制作404跳转页的简单实现方法
2016/09/22 PHP
php上传excel表格并获取数据
2017/04/27 PHP
js prototype截取字符串函数
2010/04/01 Javascript
基于JQuery实现异步刷新的代码(转载)
2011/03/29 Javascript
使用js 设置url参数
2013/07/08 Javascript
js获取IP和PcName(IE)在vs中可用
2013/08/02 Javascript
javascript中声明函数的方法及调用函数的返回值
2014/07/22 Javascript
JavaScript动态检验密码强度的实现方法
2016/11/09 Javascript
JS设置时间无效问题的解决办法
2017/02/18 Javascript
js实现功能比较全面的全选和多选
2017/03/02 Javascript
vue2.0中click点击当前li实现动态切换class
2017/06/21 Javascript
Vue使用高德地图搭建实时公交应用功能(地图 + 附近站点+线路详情 + 输入提示+换乘详情)
2018/05/16 Javascript
Babel 入门教程学习笔记
2018/06/13 Javascript
React Component存在的几种形式详解
2018/11/06 Javascript
关于React动态加载路由处理的相关问题
2019/01/07 Javascript
Vue的全局过滤器和私有过滤器的实现
2020/04/20 Javascript
html中创建并调用vue组件的几种方法汇总
2020/11/17 Javascript
在Python中使用PIL模块对图片进行高斯模糊处理的教程
2015/05/05 Python
python利用thrift服务读取hbase数据的方法
2018/12/27 Python
Django框架会话技术实例分析【Cookie与Session】
2019/05/24 Python
Python使用type关键字创建类步骤详解
2019/07/23 Python
python实现爬虫抓取小说功能示例【抓取金庸小说】
2019/08/09 Python
python的移位操作实现详解
2019/08/21 Python
python集合的创建、添加及删除操作示例
2019/10/08 Python
pytorch逐元素比较tensor大小实例
2020/01/03 Python
深入理解Tensorflow中的masking和padding
2020/02/24 Python
numpy的Fancy Indexing和array比较详解
2020/06/11 Python
python+opencv3.4.0 实现HOG+SVM行人检测的示例代码
2021/01/28 Python
施华洛世奇澳大利亚官网:SWAROVSKI澳大利亚
2017/01/06 全球购物
加拿大国民体育购物网站:National Sports
2018/11/04 全球购物
招商专员岗位职责
2014/02/08 职场文书
关于flex 上下文中自动 margin的问题(完整例子)
2021/05/20 HTML / CSS