vue中是怎样监听数组变化的


Posted in Javascript onOctober 24, 2020

我们知道通过Object.defineProperty()劫持数组为其设置getter和setter后,调用的数组的push、splice、pop等方法改变数组元素时并不会触发数组的setter,这就会造成使用上述方法改变数组后,页面上并不能及时体现这些变化,也就是数组数据变化不是响应式的(对上述不了解的可以参考这篇文章)。但实际用vue开发时,对于响应式数组,使用push、splice、pop等方法改变数组时,页面会及时体现这种变化,那么vue中是如何实现的呢?

通过vue源码可以看出,vue重写了数组的push、splice、pop等方法。

// src/core/observer/array.js

// 获取数组的原型Array.prototype,上面有我们常用的数组方法
const arrayProto = Array.prototype
// 创建一个空对象arrayMethods,并将arrayMethods的原型指向Array.prototype
export const arrayMethods = Object.create(arrayProto)

// 列出需要重写的数组方法名
const methodsToPatch = [
 'push',
 'pop',
 'shift',
 'unshift',
 'splice',
 'sort',
 'reverse'
]
// 遍历上述数组方法名,依次将上述重写后的数组方法添加到arrayMethods对象上
methodsToPatch.forEach(function (method) {
 // 保存一份当前的方法名对应的数组原始方法
 const original = arrayProto[method]
 // 将重写后的方法定义到arrayMethods对象上,function mutator() {}就是重写后的方法
 def(arrayMethods, method, function mutator (...args) {
  // 调用数组原始方法,并传入参数args,并将执行结果赋给result
  const result = original.apply(this, args)
  // 当数组调用重写后的方法时,this指向该数组,当该数组为响应式时,就可以获取到其__ob__属性
  const ob = this.__ob__
  let inserted
  switch (method) {
   case 'push':
   case 'unshift':
    inserted = args
    break
   case 'splice':
    inserted = args.slice(2)
    break
  }
  if (inserted) ob.observeArray(inserted)
  // 将当前数组的变更通知给其订阅者
  ob.dep.notify()
  // 最后返回执行结果result
  return result
 })
})

从上面可以看出array.js中重写了数组的push、pop、shift、unshift、splice、sort、reverse七种方法,重写方法在实现时除了将数组方法名对应的原始方法调用一遍并将执行结果返回外,还通过执行ob.dep.notify()将当前数组的变更通知给其订阅者,这样当使用重写后方法改变数组后,数组订阅者会将这边变化更新到页面中。

重写完数组的上述7种方法外,我们还需要将这些重写的方法应用到数组上,因此在Observer构造函数中,可以看到在监听数据时会判断数据类型是否为数组。当为数组时,如果浏览器支持__proto__,则直接将当前数据的原型__proto__指向重写后的数组方法对象arrayMethods,如果浏览器不支持__proto__,则直接将arrayMethods上重写的方法直接定义到当前数据对象上;当数据类型为非数组时,继续递归执行数据的监听。

// src/core/observer/index.js
export class Observer {
 ...
 constructor (value: any) {
  this.value = value
  this.dep = new Dep()
  this.vmCount = 0
  def(value, '__ob__', this)
  if (Array.isArray(value)) {
   if (hasProto) {
    protoAugment(value, arrayMethods)
   } else {
    copyAugment(value, arrayMethods, arrayKeys)
   }
   this.observeArray(value)
  } else {
   this.walk(value)
  }
 }
 ...
}
function protoAugment (target, src: Object) {
 /* eslint-disable no-proto */
 target.__proto__ = src
 /* eslint-enable no-proto */
}
function copyAugment (target: Object, src: Object, keys: Array<string>) {
 for (let i = 0, l = keys.length; i < l; i++) {
  const key = keys[i]
  def(target, key, src[key])
 }
}

经过上述处理后,对于数组,当我们调用其方法处理数组时会按照如下原型链来获取数组方法:

vue中是怎样监听数组变化的

对于响应式数组,当浏览器支持__proto__属性时,使用push等方法时先从其原型arrayMethods上寻找push方法,也就是重写后的方法,处理之后数组的变化会通知到其订阅者,更新页面,当在arrayMethods上查询不到时会向上在Array.prototype上查询;当浏览器不支持__proto__属性时,使用push等方法时会先从数组自身上查询,如果查询不到会向上再Array.prototype上查询。

对于非响应式数组,当使用push等方法时会直接从Array.prototype上查询。

值得一提的是源码中通过判断浏览器是否支持__proto__来分别使用protoAugment和copyAugment 方法将重写后的数组方法应用到数组中,这是因为对于IE10及以下的IE浏览器是不支持__proto__属性的:

上述截图参考于Vue源码解析五——数据响应系统

结论:

在将数组处理成响应式数据后,如果使用数组原始方法改变数组时,数组值会发生变化,但是并不会触发数组的setter来通知所有依赖该数组的地方进行更新,为此,vue通过重写数组的某些方法来监听数组变化,重写后的方法中会手动触发通知该数组的所有依赖进行更新。

如果我的内容能对你有所帮助,我就很开心啦!

以上就是vue中是怎样监听数组变化的的详细内容,更多关于vue 监听数组变化的资料请关注三水点靠木其它相关文章!

Javascript 相关文章推荐
由document.body和document.documentElement想到的
Apr 13 Javascript
3Z版基于jquery的图片复选框(asp.net+jquery)
Apr 12 Javascript
javascript 异步页面查询实现代码(asp.net)
May 26 Javascript
jQuery实现带遮罩层效果的blockUI弹出层示例【附demo源码下载】
Sep 14 Javascript
JS基于递归实现网页版计算器的方法分析
Dec 20 Javascript
jQuery实现炫丽的3d旋转星空效果
Jul 04 jQuery
关于React动态加载路由处理的相关问题
Jan 07 Javascript
Vue组件的使用及个人理解与介绍
Feb 09 Javascript
javascript+HTML5 canvas绘制时钟功能示例
May 15 Javascript
全面分析JavaScript 继承
May 30 Javascript
seajs和requirejs模块化简单案例分析
Aug 26 Javascript
关于Layui Table隐藏列问题
Sep 16 Javascript
JSON stringify方法原理及实例解析
Oct 23 #Javascript
vue中使用腾讯云Im的示例
Oct 23 #Javascript
vue中使用router全局守卫实现页面拦截的示例
Oct 23 #Javascript
vue使用video插件vue-video-player详解
Oct 23 #Javascript
vue-video-player视频播放器使用配置详解
Oct 23 #Javascript
Javascript实现贪吃蛇小游戏(含详细注释)
Oct 23 #Javascript
Vue toFixed保留两位小数的3种方式
Oct 23 #Javascript
You might like
TP5框架实现上传多张图片的方法分析
2020/03/29 PHP
JavaScript入门教程(1) 什么是JS
2009/01/31 Javascript
jquery.validate使用攻略 第三部
2010/07/01 Javascript
关闭ie窗口清除Session的解决方法
2014/01/10 Javascript
jquery获取复选框被选中的值
2014/03/22 Javascript
js实现人才网站职位选择功能的方法
2015/08/14 Javascript
jquery实现简单的全选和反选功能
2016/01/02 Javascript
jQuery Select下拉框操作小结(推荐)
2016/07/22 Javascript
Angular外部使用js调用Angular控制器中的函数方法或变量用法示例
2016/08/05 Javascript
javascript使用btoa和atob来进行Base64转码和解码
2017/03/20 Javascript
JSON对象转化为字符串详解
2017/08/11 Javascript
基于D3.js实现时钟效果
2018/07/17 Javascript
小程序云开发部署攻略(图文教程)
2018/10/30 Javascript
js使用文档就绪函数动态改变页面内容示例【innerHTML、innerText】
2019/11/07 Javascript
如何利用node.js开发一个生成逐帧动画的小工具
2019/12/01 Javascript
vue-openlayers实现地图坐标弹框效果
2020/09/24 Javascript
JQuery Ajax如何实现注册检测用户名
2020/09/25 jQuery
跟老齐学Python之print详解
2014/09/28 Python
Python中使用第三方库xlrd来读取Excel示例
2015/04/05 Python
使用Python的turtle模块画图的方法
2017/11/15 Python
浅谈Python中重载isinstance继承关系的问题
2018/05/04 Python
python判断一个集合是否为另一个集合的子集方法
2018/05/04 Python
python基于opencv 实现图像时钟
2021/01/04 Python
HTML5中判断横屏竖屏的方法(移动端)
2016/08/04 HTML / CSS
阿迪达斯奥地利官方商城:adidas.at
2016/10/16 全球购物
Exoticca英国:以最优惠的价格提供豪华异国情调旅行
2018/10/18 全球购物
女子锻炼服装和瑜伽服装:Splits59
2019/03/04 全球购物
英国打印机墨盒销售网站:Ink Factory
2019/10/07 全球购物
描述内存分配方式以及它们的区别
2016/10/15 面试题
班长自荐书范文
2014/02/11 职场文书
2014年党务公开方案
2014/05/08 职场文书
协议书范文
2015/01/27 职场文书
2015年国庆晚会主持词
2015/07/01 职场文书
食品卫生管理制度
2015/08/06 职场文书
python opencv通过按键采集图片源码
2021/05/20 Python
如何理解python接口自动化之logging日志模块
2021/06/15 Python