You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
constobj={foo: "foo",bar: "bar"}observe(obj)deleteobj.foo// no okobj.jar='xxx'// no ok
当我们对一个数组进行监听的时候,并不那么好使了
constarrData=[1,2,3,4,5];arrData.forEach((val,index)=>{defineProperty(arrData,index,val)})arrData.push()// no okarrData.pop()// no okarrDate[0]=99// ok
面试官:Vue3.0里为什么要用 Proxy API 替代 defineProperty API ?
一、Object.defineProperty
定义:
Object.defineProperty()
方法会直接在一个对象上定义一个新属性,或者修改一个对象的现有属性,并返回此对象为什么能实现响应式
通过
defineProperty
两个属性,get
及set
属性的 getter 函数,当访问该属性时,会调用此函数。执行时不传入任何参数,但是会传入 this 对象(由于继承关系,这里的this并不一定是定义该属性的对象)。该函数的返回值会被用作属性的值
属性的 setter 函数,当属性值被修改时,会调用此函数。该方法接受一个参数(也就是被赋予的新值),会传入赋值时的 this 对象。默认为 undefined
下面通过代码展示:
定义一个响应式函数
defineReactive
调用
defineReactive
,数据发生变化触发update
方法,实现数据响应式在对象存在多个
key
情况下,需要进行遍历如果存在嵌套对象的情况,还需要在
defineReactive
中进行递归当给
key
赋值为对象的时候,还需要在set
属性中进行递归上述例子能够实现对一个对象的基本响应式,但仍然存在诸多问题
现在对一个对象进行删除与添加属性操作,无法劫持到
当我们对一个数组进行监听的时候,并不那么好使了
可以看到数据的
api
无法劫持到,从而无法实现数据响应式,所以在
Vue2
中,增加了set
、delete
API,并且对数组api
方法进行一个重写还有一个问题则是,如果存在深层的嵌套对象关系,需要深层的进行监听,造成了性能的极大问题
小结
API
方法无法监听到二、proxy
Proxy
的监听是针对一个对象的,那么对这个对象的所有操作会进入监听操作,这就完全可以代理所有属性了在
ES6
系列中,我们详细讲解过Proxy
的使用,就不再述说了下面通过代码进行展示:
定义一个响应式方法
reactive
测试一下简单数据的操作,发现都能劫持
再测试嵌套对象情况,这时候发现就不那么 OK 了
如果要解决,需要在
get
之上再进行一层代理三、总结
Object.defineProperty
只能遍历对象属性进行劫持Proxy
直接可以劫持整个对象,并返回一个新对象,我们可以只操作新的对象达到响应式目的Proxy
可以直接监听数组的变化(push
、shift
、splice
)Proxy
有多达13种拦截方法,不限于apply
、ownKeys
、deleteProperty
、has
等等,这是Object.defineProperty
不具备的正因为
defineProperty
自身的缺陷,导致Vue2
在实现响应式过程需要实现其他的方法辅助(如重写数组方法、增加额外set
、delete
方法)Proxy
不兼容IE,也没有polyfill
,defineProperty
能支持到IE9参考文献
The text was updated successfully, but these errors were encountered: