参考文章:https://www.cnblogs.com/liugang-vip/p/5616484.html

概述

事件委托也叫事件代理,JavaScript高级程序设计上讲:事件委托就是利用事件冒泡,只指定一个事件处理程序,就可以管理某一类型的所有事件。

有三个同事预计会在周一收到快递。为签收快递,有两种办法:一是三个人在公司门口等快递;二是委托给前台MM代为签收。现实当中,我们大都采用委托的方案。前台MM收到快递后,她会判断收件人是谁,然后按照收件人的要求签收,甚至代为付款。这种方案还有一个优势,那就是即使公司里来了新员工(不管多少),前台MM也会在收到寄给新员工的快递后核实并代为签收。

这里其实还有2层意思的:

第一,现在委托前台的同事是可以代为签收的,即程序中的现有的dom节点是有事件的;

第二,新员工也是可以被前台MM代为签收的,即程序中新添加的dom节点也是有事件的。

为什么要用事件委托

一般来说,dom需要有事件处理程序,我们都会直接给它设事件处理程序就好了,那如果是很多的dom需要添加事件处理呢?

比如我们有100个li,每个li都有相同的click点击事件,可能我们会用for循环的方法,来遍历所有的li,然后给它们添加事件,那这么做会存在什么影响呢?

在JavaScript中,添加到页面上的事件处理程序数量将直接关系到页面的整体运行性能,因为需要不断的与dom节点进行交互,访问dom的次数越多,引起浏览器重绘与重排的次数也就越多,就会延长整个页面的交互就绪时间。

性能优化的主要思想之一就是减少DOM操作,如果要用事件委托,就会将所有的操作放到js程序里面,与dom的操作就只需要交互一次,这样就能大大的减少与dom的交互次数,提高性能;

每个函数都是一个对象,是对象就会占用内存,对象越多,内存占用率就越大,自然性能就越差了(内存不够用,是硬伤,哈哈),比如上面的100个li,就要占用100个内存空间,如果是1000个,10000个呢,那只能说呵呵了,如果用事件委托,那么我们就可以只对它的父级(如果只有一个父级)这一个对象进行操作,这样我们就需要一个内存空间就够了,是不是省了很多,自然性能就会更好。

事件委托的原理

事件委托是利用事件的冒泡原理来实现的,有这样一个机制,那么我们给最外面的div加点击事件,那么里面的ul,li,a做点击事件的时候,都会冒泡到最外层的div上,所以都会触发,这就是事件委托,委托它们父级代为执行事件

浏览器默认的为冒泡型事件触发机制(代码不用刻意设置)

事件委托怎么实现

先看看我们传统思维的做法(不提倡)

window.onload = function(){var oul = document.getElementById('ul1')var ali = document.getElementsByTagName('li')for(var i=0;i<ali.length;i++){ali[i].onclick = function(){alert(111)}}}
复制代码
<ul id="ul1"><li>111</li><li>222</li><li>333</li><li>444</li></ul>
复制代码

用事件委托的方式

window.onload = function(){var oul = document.getElementById('ul1')oul.onclick = function(){alert(111)}}
复制代码

这里用父级ul做事件处理,当li被点击时,由于冒泡原理,事件就会冒泡到ul上,因为ul上有点击事件,所以事件就会触发

当然,这里当点击整个ul的时候,也是会触发的

那么问题就来了,如果我想让事件代理的效果跟直接给节点的事件效果一样怎么办,比如说只有点击li才会触发,不怕,我们有绝招:

事件处理效果一样的:

//只对目标标签触发事件,外层点击没反应window.onload = function(){var oul = document.getElementById('ul1')oul.onclick = function(ev){var ev = ev || window.eventvar target = ev.target || ev.srcElementif(target.nodeName.toLowerCase() == 'li'){alert('弹出 '+target.innerHTML)}}}
复制代码

事件处理效果不一样的:

<div id="box"><input type="button" id="add" value="添加"/><input type="button" id="move" value="移动"/><input type="button" id="delete" value="删除"/><input type="button" id="select" value="选择"/></div>
复制代码
window.onload = function(){var box = document.getElementById('box')box.onclick = function(ev){var ev = ev || window.eventvar target = ev.target || ev.srcElementif(target.nodeName.toLowerCase() == 'input'){switch(target.id){case 'add':alert('添加');break;case 'move':alert('移动');break;case 'delete':alert('删除');break;case 'select':alert('选择');break;}}}}
复制代码

用事件委托就可以只用一次dom操作就能完成所有的效果,比上面的性能肯定是要好一些的

之前讲的都是document加载完成的现有dom节点下的操作,那么如果是新增的节点,新增的节点会有事件吗?也就是说,一个新员工来了,他能收到快递吗?

正常的添加节点的方法:

<input type="button" id="add" value="添加"/><ul id="ul1"><li>111</li><li>222</li><li>333</li><li>444</li></ul>
复制代码
//新增节点的,还能触发事件嘛?window.onload = function(){var oul = document.getElementById('ul1')var ali = document.getElementsByTagName('li')var btn = document.getElementById('add')var num = 4//鼠标移入li上变红,移出变白for(var i=0;i<ali.length;i++){ali[i].onmouseover = function(){this.style.background = 'red'}ali[i].onmouseout = function(){this.style.background = '#FFFFFF'}}//添加新节点btn.onclick = function(){num++var oli = document.createElement('li')oli.innerHTML = 111*numoul.appendChild(oli)}}
复制代码

你会发现,新增的li是没有事件的,说明添加子节点的时候,事件没有一起添加进去,这不是我们想要的结果,那怎么做呢?

一般的解决方案会是这样,将for循环用一个函数包起来,命名为mHover,如下:

//新增节点的,还能触发事件嘛?window.onload = function(){var oul = document.getElementById('ul1')var ali = document.getElementsByTagName('li')var btn = document.getElementById('add')var num = 4//鼠标移入li上变红,移出变白function mHover(){for(var i=0;i<ali.length;i++){ali[i].onmouseover = function(){this.style.background = 'red'}ali[i].onmouseout = function(){this.style.background = '#FFFFFF'}}}mHover()//添加新节点btn.onclick = function(){num++var oli = document.createElement('li')oli.innerHTML = 111*numoul.appendChild(oli)mHover()}}
复制代码

虽然功能实现了,看着还挺好,但实际上无疑是又增加了一个dom操作,在优化性能方面是不可取的,那么有事件委托的方式,能做到优化吗?

//用事件委托window.onload = function(){var oul = document.getElementById('ul1')var btn = document.getElementById('add')var num = 4//鼠标移入li上变红,移出变白oul.onmouseover = function(ev){var ev = ev||window.eventvar target = ev.target || ev.srcElementif(target.nodeName.toLowerCase() == 'li'){target.style.background = 'red'}}oul.onmouseout = function(ev){var ev = ev||window.eventvar target = ev.target || ev.srcElementif(target.nodeName.toLowerCase() == 'li'){target.style.background = '#FFFFFF'}}//添加新节点btn.onclick = function(){num++var oli = document.createElement('li')oli.innerHTML = 111*numoul.appendChild(oli)}}
复制代码

看,上面是用事件委托的方式,新添加的子元素是带有事件效果的,我们可以发现,当用事件委托的时候,根本就不需要去遍历元素的子节点,只需要给父级元素添加事件就好了,其他的都是在js里面的执行,这样新增的节点也有事件,这样可以大大的减少dom操作,这才是事件委托的精髓所在。

js中的事件委托或是事件代理详解相关推荐

  1. html offsetwidth 字符串宽度,基于js中style.width与offsetWidth的区别(详解)

    作为一个初学者,经常会遇到在获取某一元素的宽度(高度.top值...)时,到底是用 style.width还是offsetWidth的疑惑. 1. 当样式写在行内的时候,如 时,用 style.wid ...

  2. JS中NaN、NULL、undefined、详解

    NaN:保留字(表明数据类型不是数字) undefined:对象属性或方法不存在,或声明了变量但从未赋值. 即当你使用了对象未定的属性或者未定义的方法时或当你声明一个变量,但你确从未对其进行赋值,便对 ...

  3. js中childNodes为什么输出#text:childNodes详解

    转载原创地址:http://www.cnblogs.com/Jersen/p/4908943.html 定义和用法 childNodes 属性返回节点的子节点集合,以 NodeList 对象. 提示: ...

  4. js 浅拷贝直接赋值_JS中实现浅拷贝和深拷贝的代码详解

    (一)JS中基本类型和引用类型 JavaScript的变量中包含两种类型的值:基本类型值 和 引用类型值,在内存中的表现形式在于:前者是存储在栈中的一些简单的数据段,后者则是保存在堆内存中的一个对象. ...

  5. js中的事件委托或是事件代理详解(转载)

    起因: 1.这是前端面试的经典题型,要去找工作的小伙伴看看还是有帮助的: 2.其实我一直都没弄明白,写这个一是为了备忘,二是给其他的知其然不知其所以然的小伙伴们以参考: 概述: 那什么叫事件委托呢?它 ...

  6. JS中的事件委托/事件代理详解

    起因: 1.这是前端面试的经典题型,要去找工作的小伙伴看看还是有帮助的: 2.其实我一直都没弄明白,写这个一是为了备忘,二是给其他的知其然不知其所以然的小伙伴们以参考: 概述: 那什么叫事件委托呢?它 ...

  7. JS中的事件委托 / 代理详解

    [前言] 事件委托/代理是前端面试的经典题型,要去找工作的小伙伴看看还是有帮助的 [主体] 概述: 那什么叫事件委托呢?它还有一个名字叫事件代理,JavaScript高级程序设计上讲:事件委托就是利用 ...

  8. JavaScript系列—简述JS中的事件委托和事件代理

    JS中的事件委托和事件代理 什么是事件委托? 事件委托还有一个名字叫事件代理,JS高程上讲:事件委托就是利用事件冒泡,只制定一个时间处理程序,就可以管理某一类型的所有事件.我用取快递来解释这个现象: ...

  9. JS事件委托或者事件代理原理以及实现

    事件委托(事件代理)原理:简单的说就是将事件交由别人来执行,就是将子元素的事件通过冒泡的形式交由父元素来执行. 为什么要用时间委托? 在JavaScript中,添加到页面上的事件处理程序数量将直接关系 ...

最新文章

  1. 记录JVM垃圾回收算法
  2. EM算法理解的九层境界
  3. SAP PI 适配器引擎
  4. 设计模式(创建型模式)——单例模式(Singleton)
  5. 医疗机器人等高智能医疗设备成未来发展重点领域
  6. python3数据库表关联_Django中数据库操作|python3教程|python入门|python教程
  7. 获取计算机中的硬件信息
  8. 【解决pip install 安装报错】No module named ‘_ctypes‘
  9. e.printStackTrace() 会导致锁死?
  10. LeetCode 2086. 从房屋收集雨水需要的最少水桶数(贪心)
  11. 修改sqoop存储job的默认数据库
  12. 根据TTL值判断目标主机的类型
  13. 【疾病分类】基于matlab SVM植物叶子疾病检测和分类【含Matlab源码 093期】
  14. Java docx4j 操作word 1.0
  15. DDS(Direct Digital Synthesizer)数字频率合成器
  16. LeetCode 三等分(题解+优化过程)
  17. Mac安装Linux虚拟机
  18. java实现密码重置功能_如何实施密码重置链接
  19. 【回归预测-FNN预测】基于蝙蝠算法优化前馈网络实现数据回归预测附Matlab代码
  20. Vue实现导航栏切换

热门文章

  1. SAP 更改登录界面图片
  2. SpringBoot+Vue实现前后端分离的宠物医院管理系统
  3. XP系统下MSDTC不可用解决办法
  4. [iOS] AFNetworking 的内存泄漏分析
  5. suse种运行wkhtmltopdf
  6. 网络安全专业术语对照
  7. ArangoDB压测小记
  8. linux下通过SD卡烧录.bin程序
  9. 微信公众号身份证OCR识别和验真|人证比对
  10. springboot实现反向代理