【Vue】浅谈vue2、vue3响应式原理,vue中数组的响应式,响应式常见问题分析
创始人
2024-05-26 00:42:19
0

前言:此处响应式指的是数据响应式变化,而不是页面的响应式布局,页面的响应式布局在我的其他文章中有提到。

一、什么是vue响应式

Vue 最标志性的功能就是其低侵入性的响应式系统。组件状态都是由响应式的 JavaScript 对象组成的。当更改它们时,视图会随即自动更新。

二、vue2的响应式原理

2.1 官方解释

当你把一个普通的 JavaScript 对象传入 Vue 实例作为 data 选项,Vue 将遍历此对象所有的 property,并使用 Object.defineProperty 把这些 property 全部转为 getter/setter

2.2 大白话解释

Vue在组件和实例初始化的时候,会将data()里的数据进行数据劫持,即使用object.definepropty对数据做处理。被劫持处理过后的数据会有两个方法:一个叫getter,一个叫setter

  • getter是使用数据的时候触发,
  • setter是在修改数据的时候触发,触发setter的同时也触发了底层的watcher监听,通知dom修改刷新。
    在这里插入图片描述

2.3 数组的响应式

  • 对于对象会遍历它所有的属性,然后使用Object.defineProperty重写get、set方法,对对象的每个属性进行劫持。如果属性值还是对象,则会进行递归。

  • 对于数组,也会遍历它所有的元素,然后使用Object.defineProperty方法对每个元素进行劫持。然后还会重写数组原型上push、pop、shift、unshift、sort、reverse、splice七个方法。

对于数组本身,并没有像对象一样,使用Object.defineProperty对自身和下标重写get、set方法。这也就导致了我们直接通过下标赋新值或直接删除值是不能响应式。(为了性能考虑 Vue2 直接弃用了使用 Object.defineProperty 对数组进行监听的方案)

注意
对于数组我们不要以为通过下标修改数据就一定不能响应式,如果数组元素是引用数据类型,恰巧只需要修改该引用数据类型某属性,是可以直接通过下标更改的。
因为前面说了,虽然数组并没有像对象一样,使用Object.defineProperty对自身和下标重写get、set方法,但是它会遍历它所有的元素,然后使用Object.defineProperty方法对每个元素进行劫持。所以我们更改某引用数据类型的某属性是可以响应式的!!!

例如:

{data() {return {arr: [123, {name: '张三'}]}},methods: {updateArr(){this.arr[1]['name'] = '李四' // 这样是可以响应式的}}
}

2.4 特别备注

由于vue2使用Object.defineProperty方法,会重写get、set方法,提前将数据进行劫持。这也就导致了在后面给对象添加新属性和直接删除属性是(删除不会触发set方法)不能响应式。(即它只对初始对象的属性有监听作用)
鱿鱼须也知道有这个缺陷,提供了$set/$delete来帮助我们达到响应式。

2.5 文章

vue2的响应式原理官方文档
getter/setter的使用方法

三、vue3的响应式原理

3.1 官方解释

在 Vue 3 中则使用了 Proxy 来创建响应式对象,仅将getter / setter 用于 ref

3.2 大白话解释

Proxy 是 JavaScript 2015 的一个新特性。 Proxy 的代理是针对整个对象的,而不是对象的某个属性,因此不同于 Object.defineProperty 的必须遍历对象每个属性, Proxy 只需要做一层代理就可以监听同级结构下的所有属性变化(包括:属性值的读写,属性的增加,属性的删除等),当然对于深层结构,递归还是需要进行的。此外 Proxy支持代理数组的变化。

Proxy 本质上属于元编程非破坏性数据劫持,在原对象的基础上进行了功能的衍生而又不影响原对象,符合松耦合高内聚的设计理念。

3.3 特别备注

在IE11以下的浏览器都不兼容,所以如果使用 Vue3 开发一个单页应用的项目,需要考虑到兼容性问题,需要我们做额外的很多操作,才能使得IE11 以下的版本能够兼容。

3.4 文章

vue3的响应式原理官方文档
ES6的proxy代理的使用方法

四、常见响应式问题

4.1 vue2中数据改变但是视图未更新

1、Vue.set()或this.$set()

对于已经创建的实例,Vue 不允许动态添加根级别的响应式 property。但是,可以使用 Vue.set(object, propertyName, value) 方法向嵌套对象添加响应式 property。例如:

Vue.set(vm.someObject, 'b', 2);
this.$set(this.someObject,'b',2);

2、this.$forceUpdate()强制刷新数据

迫使 Vue 实例重新渲染。注意它仅仅影响实例本身和插入插槽内容的子组件,而不是所有子组件。

3、Object.assign()解决vue2中对象新增属性不是响应式问题

有时你可能需要为已有对象赋值多个新 property,比如使用 Object.assign() _.extend()。但是,这样添加到对象上的新 property 不会触发更新。在这种情况下,你应该用原对象与要混合进去的对象的 property 一起创建一个新的对象。
将源对象和要加入属性的对象合并为一个新的对象赋值给要响应式的对象(源对象),vue会将源对象重新触发一遍Object.defineProperty依赖收集。

this.msg = Object.assign({}, this.msg, { age: 18 }); //(响应式)

未完待续…
在这里插入图片描述

相关内容

热门资讯

华谊公司旗下艺人有哪些优秀艺人... 华谊公司旗下艺人有哪些优秀艺人?华谊兄弟旗下的艺人多达四十多人,但有名的不是很多华谊旗下有很多很不错...
为何18岁男子脑死亡火化前妈妈... 为何18岁男子脑死亡火化前妈妈大喊别离开下一秒突然复活?我觉得应该是哪里出现了纰漏,这名男子并没有死...
关于宇宙的科普电影 关于宇宙的科普电影银河系漫游指南 绝对是是吧,《旅行到宇宙边缘》旅行到宇宙边缘
北京工商大学会计专业考研科目都... 北京工商大学会计专业考研科目都有什么?专业复习用书都用什么?初始考802管理学指定参考书是周三多《管...
作者为什么要写丁香结? 作者为什么要写丁香结?“丁香结”的本义是小小的花苞圆圆的,鼓鼓的,恰如衣襟上的盘花扣。象征意义是人生...
原核生物和真核生物的区别 原核生物和真核生物的区别真核有复杂的内膜系统(细胞器),原核生物无真核有细胞核,原核无成型的细胞核基...
怎样不吃药摆脱抑郁症 怎样不吃药摆脱抑郁症以前靠心理治疗,但是后面有一段时期加重了。但是药的副作用太大,可以不吃药吗?想开...
诺曼底登陆的作用和对二战的意义 诺曼底登陆的作用和对二战的意义改变了二战的战局,加快盟军胜利的到来。
怎么样才能学习学得“细”呢? 怎么样才能学习学得“细”呢?细是注意细节地方,平时做题多思考,尽量全面一些,不要学习时觉的理解了就完...
去年几滴相思水,化做树下种花泪... 去年几滴相思水,化做树下种花泪,谁家少年踏春来,折下枝头红玫瑰!这首诗的意思是什么?暗示爱意。指思念...
小花仙找主人! 小花仙找主人!我只要你帮我照顾花朵,每月一个礼物可以吗?我有额外条件:一个星期星期一,三,五一定要上...
手机在信号不好的时候 对方打不... 手机在信号不好的时候 对方打不进来 能设置来电转移吗rt 是否需要开通什么业务 怎么办理不需要去移...
与看透生死或看透人生有关的歌曲... 与看透生死或看透人生有关的歌曲 比如悟我觉得三国演义主题曲有一种世态苍爽的感觉 ,得意的笑 潇洒走...
面临就业,该如何帮助女朋友度过... 面临就业,该如何帮助女朋友度过心里难关?不要说话,这种时候,什么语言都是多余和反作用的,要求对方走出...
八仙 全传 八仙 全传网上的视频有2个版本..一个是有60集.一个有40集..,你直接去查大结局就好了.没有这回...
正在追求一个女孩子,我告白了,... 正在追求一个女孩子,我告白了,但是。女生优先进,请求各位好心人,我真得很爱这个女孩他对你是在观察期不...
什么是TF卡? 什么是TF卡?什么是TF卡,还有SD卡??SD卡(TF卡)即是手机储存的扩展卡,主要用来扩展手机的储...
杨幂许凯路透,两人将合作哪部剧... 杨幂许凯路透,两人将合作哪部剧呢?路透当中的照片是杨幂和许凯两人为主演所演的《爱的二八定律》,非常的...
中国十大产业调查研究公司有哪些... 中国十大产业调查研究公司有哪些?根据自己的经验,以下信息有点价值,仅供参考:
有心人是什么意思 有心人是什么意思意中人中意人有心人是一个汉语词汇,解释为对某个事物或人等等,有一种感觉。例句:想不到...