浅析javascript中的事件代理


Posted in Javascript onNovember 06, 2015

本文的主要内容是根据前不久面试某家公司Web前端开发岗位,面试时做的一道数组去重问题的解题思路进行整理的,分享给大家。

题目本身很简单:一个ul中有一千个li,如何给这一千个li绑定一个鼠标点击事件,当鼠标点击时alert出这个li的内容和li的位置坐标xy,

<ul id="ulItem">
 <li id="li1">1</li>
 <li id="li2">2</li>
 <li id="li3">3</li>
 ...
 <li id="li1000">1000</li>
</ul>

需要考虑到浏览器兼容性、事件冒泡、效率等问题。看到问题后我就直接在纸上写下了如下答案:

var ulItem = document.getElementById("ulItem");
var lis = document.getElementsByTagName("li");
for(var i=0; i<lis.length; i++){
 lis[i].onclick = function(){
 alert("内容:"+this.innerHTML);
 alert("位置:"+getElementPosition(this).x+","+getElementPosition(this).y;
 }
}
function getElementPosition(e){
 var x=0,y=0;
 while(e != null){
 x += e.offsetLeft;
 y += e.offsetTop;
 e = e.offsetParent;
 }<br>return {x:x, y:y};
}

面试结果:写完了又看了一遍感觉没必要考虑兼容性、事件冒泡啊。效率的话,想了想,也想不出怎么提升了,就这样给面试官看了。面试官人也挺好的,他看了之后说:你并没有考虑到我说的重点啊,你这样1000次循环添加点击事件效率是很低的。然后就跟我讲了利用事件冒泡的特性,来提高效率,即事件代理(ps:以前做项目有遇到过要阻止事件冒泡的时候,但利用事件冒泡特性提高效率却还完全不知道)。听了面试官讲的涨了姿势,回来后自己也上网查了一下,现在自己再总结下当做记录自己学习的过程吧:

事件代理(Event Delegation),又称之为事件委托。是 JavaScript 中常用绑定事件的常用技巧。顾名思义,“事件代理”即是把原本需要绑定的事件委托给父元素,让父元素担当事件监听的职务。
为什么要这么做?众所周知,DOM操作是十分消耗性能的,所以重复的事件绑定简直是性能杀手。而事件代理的核心思想,就是通过尽量少的绑定,去监听尽量多的事件。程序猿的事,没代码说个J8,下面贴出代码:

var ulItem = document.getElementById("ulItem");
ulItem.onclick = function(e){
 e = e || window.event;//这一行和下一行是为了兼容IE8以及之前版本
 var target = e.target || e.srcElement;
 if(target.tagName.toLowerCase() === "li"){
 alert(target.innerHTML);
 alert("位置为:"+getElementPosition(target).x+","+getElementPosition(target).y);
 }
}
function getElementPosition(e){
 var x=0,y=0;
 while(e != null){
 x += e.offsetLeft;
 y += e.offsetTop;
 e = e.offsetParent;
 }
return {x:x, y:y};
}

嗯,现在代码去掉了for循环,提高了效率,也有了兼容性方面的处理,感觉这个答案应该可以了吧。上面说的也就是为了一道笔试题,下面就再本着学术研究的思想说说事件代理:

在传统的事件处理中,你按照需要为每一个元素添加或者是删除事件处理器。然而,事件处理器将有可能导致内存泄露或者是性能下降——你用得越多这种风险就越大。JavaScript事件代理则是一种简单的技巧,通过它你可以把事件处理器添加到一个父级元素上,这样就避免了把事件处理器添加到多个子级元素上。事件代理用到了两个在JavaSciprt事件中常被忽略的特性:事件冒泡以及目标元素。当一个元素上的事件被触发的时候,比如说鼠标点击了一个按钮,同样的事件将会在那个元素的所有祖先元素中被触发。这一过程被称为事件冒泡;这个事件从原始元素开始一直冒泡到DOM树的最上层。任何一个事件的目标元素都是最开始的那个元素,在我们的这个例子中也就是按钮,并且它在我们的事件对象中以属性的形式出现。使用事件代理,我们可以把事件处理器添加到一个元素上,等待一个事件从它的子级元素里冒泡上来,并且可以得知这个事件是从哪个元素开始的。

关于事件代理,今天也是初次接触,就先写到这吧,希望对大家的学习有所帮助。

Javascript 相关文章推荐
用javascript实现的支持lrc歌词的播放器
May 17 Javascript
使用JQuery进行跨域请求
Jan 25 Javascript
JavaScript排序算法之希尔排序的2个实例
Apr 04 Javascript
JavaScript SHA512&amp;SHA256加密算法详解
Aug 11 Javascript
jQuery form插件的使用之处理server返回的JSON, XML,HTML数据
Jan 26 Javascript
手机端js和html5刮刮卡效果
Sep 29 Javascript
touch.js 拖动、缩放、旋转 (鼠标手势)功能代码
Feb 04 Javascript
vue上传图片组件编写代码
Jul 26 Javascript
打通前后端构建一个Vue+Express的开发环境
Jul 17 Javascript
Vue 配合eiement动态路由,权限验证的方法
Sep 26 Javascript
JS 验证码功能的三种实现方式
Nov 26 Javascript
使用Typescript和ES模块发布Node模块的方法
May 25 Javascript
详解javascript中的事件处理
Nov 06 #Javascript
jQuery插件实现静态HTML验证码校验
Nov 06 #Javascript
jQuery Real Person验证码插件防止表单自动提交
Nov 06 #Javascript
jQuery实现非常实用漂亮的select下拉菜单选择效果
Nov 06 #Javascript
javascript如何实现暂停功能
Nov 06 #Javascript
JavaScript实现带缓冲效果的随屏滚动漂浮广告代码
Nov 06 #Javascript
jQuery实现可关闭固定于底(顶)部的工具条菜单效果
Nov 06 #Javascript
You might like
smtp邮件发送一例
2006/10/09 PHP
兼容性最强的PHP生成缩略图的函数代码(修改版)
2011/01/18 PHP
PHP面向接口编程 耦合设计模式 简单范例
2011/03/23 PHP
php通过session防url攻击方法
2014/12/10 PHP
PHP中PDO的事务处理分析
2016/04/07 PHP
php解析xml 的四种简单方法(附实例)
2016/07/11 PHP
php实现微信扫码自动登陆与注册功能
2016/09/22 PHP
关于PHP虚拟主机概念及如何选择稳定的PHP虚拟主机
2018/11/20 PHP
js实现文字在按钮上滚动的方法
2015/08/20 Javascript
jQuery如何使用自动触发事件trigger
2015/11/29 Javascript
jQuery实现拖拽页面元素并将其保存到cookie的方法
2016/06/12 Javascript
浅谈js中test()函数在正则中的使用
2016/08/19 Javascript
郁闷!ionic中获取ng-model绑定的值为undefined如何解决
2016/08/27 Javascript
利用JS实现点击按钮后图片自动切换的简单方法
2016/10/24 Javascript
JS 实现 ajax 异步浏览器兼容问题
2017/01/21 Javascript
Nodejs基于LRU算法实现的缓存处理操作示例
2017/03/17 NodeJs
JS检测是否可以访问公网服务器功能代码
2017/06/19 Javascript
NVM安装nodejs的方法实用步骤
2019/01/16 NodeJs
超简单的微信小程序轮播图
2019/11/22 Javascript
json_decode 索引为数字时自动排序问题解决方法
2020/03/28 Javascript
jQuery实现图片切换效果
2020/10/19 jQuery
[46:12]完美世界DOTA2联赛循环赛 DM vs Matador BO2第一场 11.04
2020/11/04 DOTA
python提示No module named images的解决方法
2014/09/29 Python
使用Python编写提取日志中的中文的脚本的方法
2015/04/30 Python
使用python在本地电脑上快速处理数据
2017/06/22 Python
python复制列表时[:]和[::]之间有什么区别
2018/10/16 Python
python解压TAR文件至指定文件夹的实例
2019/06/10 Python
如何在Django中使用聚合的实现示例
2020/03/23 Python
PUMA官方商城:世界领先的运动品牌之一
2016/11/16 全球购物
澳大利亚在线性感内衣商店:Fantasy Lingerie
2021/02/07 全球购物
特色冷饮店创业计划书
2014/01/28 职场文书
计算机通信专业推荐信
2014/02/22 职场文书
2014年节能减排工作总结
2014/12/06 职场文书
童年读书笔记
2015/06/26 职场文书
2016年清明节红领巾广播稿
2015/12/17 职场文书
超级实用的公文标题大全!
2019/07/19 职场文书