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获得keycode的示例代码
Dec 30 Javascript
SeaJS入门教程系列之SeaJS介绍(一)
Mar 03 Javascript
js继承call()和apply()方法总结
Dec 08 Javascript
jQuery中removeData()方法用法实例
Dec 27 Javascript
JavaScript获得页面base标签中url的方法
Apr 03 Javascript
使用AngularJS创建单页应用的编程指引
Jun 19 Javascript
jQuery实现简单的DIV拖动效果
Feb 19 Javascript
总结Javascript中数组各种去重的方法
Oct 04 Javascript
从零开始做一个pagination分页组件
Mar 15 Javascript
vue实现微信分享朋友圈,发送朋友的示例讲解
Feb 10 Javascript
浅谈Vue 性能优化之深挖数组
Dec 11 Javascript
Django+vue跨域问题解决的详细步骤
Jan 20 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
输出控制类
2006/10/09 PHP
php生成随机密码的三种方法小结
2010/09/04 PHP
php 表单提交大量数据发生丢失的解决方法
2014/03/03 PHP
PHP生成随机数的方法实例分析
2015/01/22 PHP
CI框架AR数据库操作常用函数总结
2016/11/21 PHP
javascript语句中的CDATA标签的意义
2007/05/09 Javascript
JS的document.all函数使用示例
2013/12/30 Javascript
JavaScript实现跨浏览器的添加及删除事件绑定函数实例
2015/08/04 Javascript
jQuery实现图片走马灯效果的原理分析
2016/01/16 Javascript
jQuery使用cookie与json简单实现购物车功能
2016/04/15 Javascript
jQuery插件HighCharts绘制的2D堆柱状图效果示例【附demo源码下载】
2017/03/14 Javascript
微信小程序 http请求的session管理
2017/06/07 Javascript
webpack打包js文件及部署的实现方法
2017/12/18 Javascript
Vue2.X 通过AJAX动态更新数据
2018/07/17 Javascript
简述pm2常用命令集合及配置文件说明
2019/05/30 Javascript
微信小程序移动拖拽视图-movable-view实例详解
2019/08/17 Javascript
VUE 解决mode为history页面为空白的问题
2019/11/01 Javascript
vue自定义标签和单页面多路由的实现代码
2020/05/03 Javascript
vue + node如何通过一个Txt文件批量生成MP3并压缩成Zip
2020/06/02 Javascript
如何利用JavaScript编写一个格斗小游戏
2021/01/06 Javascript
[31:33]2014 DOTA2国际邀请赛中国区预选赛 TongFu VS DT 第一场
2014/05/23 DOTA
[00:57]辉夜杯战队访谈宣传片—VG
2015/12/25 DOTA
[57:38]2018DOTA2亚洲邀请赛3月30日 小组赛A组 OpTic VS OG
2018/03/31 DOTA
利用python对Excel中的特定数据提取并写入新表的方法
2018/06/14 Python
基于Python的Post请求数据爬取的方法详解
2019/06/14 Python
pycharm快捷键汇总
2020/02/14 Python
python多维数组分位数的求取方式
2020/03/03 Python
三步解决python PermissionError: [WinError 5]拒绝访问的情况
2020/04/22 Python
2020版Python学习路线图(附学习资料)
2020/09/15 Python
小区停车场管理制度
2014/01/27 职场文书
丧事主持词大全
2014/04/02 职场文书
爱牙日活动总结
2014/08/29 职场文书
学生无故旷课检讨书
2014/09/20 职场文书
银行业务授权委托书
2014/10/10 职场文书
寒假安全保证书
2015/02/28 职场文书
大学校园招聘会感想
2015/08/10 职场文书