浅析JavaScript中的事件委托机制跟深浅拷贝


Posted in Javascript onJanuary 20, 2021

今天聊下JavaScript中的事件委托跟深浅拷贝

一、事件委托

首先呢,介绍一下事件绑定

//方法一:通过onclick
<button onclick="clickEvent()">点击</button>

<script>
function clickEvent(){
   alert("点击事件");
}
</script>

//方法二:通过addEventListener
<button id="btn_button">点击</button>

<script>
var btn = document.getElementById("btn_button");
btn.addEventListener("click", function () {
  alert("点击");
}, false);
</script>

说下其中的区别
onclick只能绑定一个事件,而addEventListener可以同时绑定多个事件

[function method1() {
  console.log("method1");
}
function method2() {
  console.log("method2");
}
function method3() {
  console.log("method3");
}
var btn = document.getElementById("btn_button");
btn.addEventListener("click", method1, false);//第三个参数是默认值,默认是冒泡,如果设置为true则是捕获
btn.addEventListener("click", method2, false);
btn.addEventListener("click", method3, false);
//最终会按顺序执行:method1 -> method2 -> method3 
btn.removeEventListener("click", method1, false);//用于移除事件


btn.onclick = method1;
btn.onclick = method2;
btn.onclick = method3;
//最终只会执行method3
btn.onclick = null;//用于移除事件

事件冒泡

<ul id="container" style="display: inline-block;">
  <li id="title1">123456</li>
  <li id="title2">qwert</li>
  <li id="title3">
    <ul>
      <li id="title3inner">title3里面的文本</li>
    </ul>
  </li>
</ul>

<script>
  var container = document.getElementById("container");
  var title1 = document.getElementById("title1");
  var title2 = document.getElementById("title2");
  var title3 = document.getElementById("title3");
  var title3inner = document.getElementById("title3inner");

  container.onclick = function (e) {
    alert("container");
  }
  title1.onclick = function (e) {
    alert("title1");
  }
  title2.onclick = function (e) {
    alert("title2");
  }
  title3.onclick = function (e) {
    alert("title3");
  }
  title3inner.onclick = function (e) {
    alert("title3inner");
  }
</script>

点击 “title3里面的文本” 会触发 3 次事件,分别 弹出
“title3inner” -> “title3” -> “container”

事件捕获

至于事件冒泡倒过来即是事件捕获,即:点击 “title3里面的文本” 会触发 3 次事件,分别 弹出
“container” -> “title3” -> “title3inner”

事件冒泡的升华

事件委托的使用

<ul id="container" style="display: inline-block;">
  <li id="title1">123456</li>
  <li id="title2">qwert</li>
  <li id="title3">QWE123</li>
</ul>

<script>
var container = document.getElementById("container");
container.onclick = function (e) {
  //console.log(e);
  if (e.target.id = "title1") {
    alert(e.target.innerText);
  }
  else if (e.target.id = "title2") {
    alert(e.target.innerText);
  }
  else if (e.target.id = "title3") {
    alert(e.target.innerText);
  }
}
</script>

至于优势嘛,便是只需注册一个点击事件,然后通过target来判断点击的具体元素。而currentTarget指的是注册事件的元素。
target是你实际中点击的元素
currentTarget是绑定事件的元素
如果不用事件委托的话,在一个列表上,每条数据后,添加一个按钮,然后给每个按钮注册一个事件,每个元素绑定一个事件,那么多的按钮,肯定会对前端性能有所影响。这时候委托肯定是最好的选择了,毕竟委托只需注册一个事件。

思考

理论上,使用事件委托确实是一种优化,只需注册一个事件,然后通过事件冒泡来实现相应的功能。
至于在Vue项目中,到底用不用事件委托呢?这里好像有个争议。
有人说在Vue中,已经帮你做过了,也有人说并没有,我没看过源码,咱也不知道。

浅析JavaScript中的事件委托机制跟深浅拷贝

总之,我是这么做的,每个按钮都绑定一个事件,然后阻止冒泡。实际中,我个人不太会考虑事件委托,感觉不是太必要的,毕竟事件并没有那么多。

二、深浅拷贝

至于深浅拷贝这块的知识,我便长话短说了,一阵见血,见血封喉!

浅析JavaScript中的事件委托机制跟深浅拷贝

let a = 3;
let b = a;
a = 4;
console.log(a);//4
console.log(b);//3
console.log(a === b);//这便是深拷贝,a和b完全是不同的变量,各自保存一个值。

let arr = [1, 2, 3, 4, 5];
let brr = arr;
arr[1] = 8;
console.log(arr[1]);//8
console.log(brr[1]);//8  为什么呢? 这便是浅拷贝,数组类型是引用类型,arr跟brr变量只是保存的引用地址,他们共同指向[1,2,3,4,5]这个数组
console.log(arr === brr);//true

实际中,如果要实现深拷贝,怎么做呢?
方法主要是一个通过递归来赋值,还有一个通过JSON.stringify与JSON.parse这两个方法来实现。
在这里使用第二种方法,毕竟最是简单,简单、暴力,也往往是最有效的解决办法。

let arr = [1, 2, 3, 4, 5];
let brr = JSON.parse(JSON.stringify(arr));
arr[1] = 8;
console.log(brr[1]);//2,这便实现了所谓的深拷贝

三、总结

正如面试官的所说,虽然以上的知识在实际工作中并不一定会用到,但你还是要知道的!还是要知道!!还是要知道的!!!

到此这篇关于浅析JavaScript中的事件委托机制跟深浅拷贝的文章就介绍到这了,更多相关js事件委托深浅拷贝内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Javascript 相关文章推荐
锋利的jQuery 要点归纳(三) jQuery中的事件和动画(上:事件篇)
Mar 24 Javascript
js获取当前日期代码适用于网页头部
Jun 27 Javascript
JavaScript判断文件上传类型的方法
Sep 02 Javascript
Javascript设计模式之观察者模式的多个实现版本实例
Mar 03 Javascript
AngularJS使用ng-Cloak阻止初始化闪烁问题的方法
Nov 03 Javascript
微信小程序购物商城系统开发系列-目录结构介绍
Nov 21 Javascript
自定义require函数让浏览器按需加载Js文件
Nov 24 Javascript
jquery easyui dataGrid动态改变排序字段名的方法
Mar 02 Javascript
Angular2中如何使用ngx-translate进行国际化
May 21 Javascript
vue2.x集成百度UEditor富文本编辑器的方法
Sep 21 Javascript
这应该是最详细的响应式系统讲解了
Jul 22 Javascript
Vue Element-ui表单校验规则实现
Jul 09 Vue.js
详解vue之自行实现派发与广播(dispatch与broadcast)
Jan 19 #Vue.js
js实现电灯开关效果
Jan 19 #Javascript
jquery实现穿梭框功能
Jan 19 #jQuery
jQuery实现穿梭框效果
Jan 19 #jQuery
element el-table表格的二次封装实现(附表格高度自适应)
Jan 19 #Javascript
JS实现纸牌发牌动画
Jan 19 #Javascript
微信小程序canvas实现签名功能
Jan 19 #Javascript
You might like
新浪微博API开发简介之用户授权(PHP基础篇)
2011/09/25 PHP
php使用codebase生成随机数
2014/03/25 PHP
PHP实现深度优先搜索算法(DFS,Depth First Search)详解
2017/09/16 PHP
yii2.0框架使用 beforeAction 防非法登陆的方法分析
2019/09/11 PHP
[IE&amp;FireFox兼容]JS对select操作
2007/01/07 Javascript
慎用 somefunction.prototype 分析
2009/06/02 Javascript
javascript 设计模式之单体模式 面向对象学习基础
2010/04/18 Javascript
基于jQuery制作迷你背词汇工具
2010/07/27 Javascript
JQuery 弹出框定位实现方法
2010/12/02 Javascript
浅析JavaScript基本类型与引用类型
2014/05/28 Javascript
Angularjs全局变量被作用域监听的正确姿势
2016/02/06 Javascript
js中scrollTop()方法和scroll()方法用法示例
2016/10/03 Javascript
JS生成一维码(条形码)功能示例
2017/01/19 Javascript
基于webpack4+vue-cli3项目实现换肤功能
2019/07/17 Javascript
微信小程序实现翻牌抽奖动画
2020/09/21 Javascript
vue使用echarts图表自适应的几种解决方案
2020/12/04 Vue.js
[01:46]2018完美盛典章节片——坚守
2018/12/17 DOTA
比较详细Python正则表达式操作指南(re使用)
2008/09/06 Python
python字符串连接的N种方式总结
2014/09/17 Python
跟老齐学Python之网站的结构
2014/10/24 Python
Python中内建函数的简单用法说明
2016/05/05 Python
Python cookbook(数据结构与算法)实现优先级队列的方法示例
2018/02/18 Python
Python 正则表达式匹配字符串中的http链接方法
2018/12/25 Python
Python爬虫之Spider类用法简单介绍
2020/08/04 Python
波兰运动鞋网上商店:Distance.pl
2020/07/30 全球购物
提高EJB性能都有哪些技巧
2012/03/25 面试题
学前教育教师求职自荐信
2013/09/22 职场文书
幼教毕业生自我鉴定
2014/01/12 职场文书
代理商会议邀请函
2014/01/27 职场文书
楼面经理岗位职责范本
2014/02/18 职场文书
房屋转让协议书范本
2014/04/11 职场文书
2014年共青团工作总结
2014/12/10 职场文书
解决pytorch-gpu 安装失败的记录
2021/05/24 Python
使用Docker容器部署rocketmq单机的全过程
2022/04/03 Servers
vue 自定义组件添加原生事件
2022/04/21 Vue.js
方法汇总:Python 安装第三方库常用
2022/04/26 Python