一、Object.defineProperty
定义:Object.defineProperty()
方法会直接在一个对象上定义一个新属性,或者修改一个对象的现有属性,并返回此对象
为什么能实现响应式? 通过defineProperty
两个属性,get
及set
- get
属性的 getter 函数,当访问该属性时,会调用此函数。执行时不传入任何参数,但是会传入 this 对象(由于继承关系,这里的this并不一定是定义该属性的对象)。该函数的返回值会被用作属性的值
- set
属性的 setter 函数,当属性值被修改时,会调用此函数。该方法接受一个参数(也就是被赋予的新值),会传入赋值时的 this 对象。默认为 undefined
定义一个响应式函数defineReactive:
调用defineReactive
,数据发生变化触发update
方法,实现数据响应式
在对象存在多个key
情况下,需要进行遍历
如果存在嵌套对象的情况,还需要在defineReactive
中进行递归
当给key
赋值为对象的时候,还需要在set
属性中进行递归
上述例子能够实现对一个对象的基本响应式,但仍然存在诸多问题
现在对一个对象进行删除与添加属性操作,无法劫持到
当我们对一个数组进行监听的时候,并不那么好使了
可以看到数据的api
无法劫持到,从而无法实现数据响应式,
所以在Vue2
中,增加了set
、delete
API,并且对数组api
方法进行一个重写
还有一个问题则是,如果存在深层的嵌套对象关系,需要深层的进行监听,造成了性能的极大问题
二、proxy
Proxy
的监听是针对一个对象的,那么对这个对象的所有操作会进入监听操作,这就完全可以代理所有属性了
下面通过代码进行展示:
定义一个响应式方法reactive
测试一下简单数据的操作,发现都能劫持再测试嵌套对象情况,这时候发现就不那么 OK 了
如果要解决,需要在get
之上再进行一层代理
三、总结
Object.defineProperty
只能遍历对象属性进行劫持
Proxy
直接可以劫持整个对象,并返回一个新对象,我们可以只操作新的对象达到响应式目的
Proxy
可以直接监听数组的变化(push
、shift
、splice
)
Proxy
有多达13种拦截方法,不限于apply
、ownKeys
、deleteProperty
、has
等等,这是Object.defineProperty
不具备的
Proxy
不兼容IE,也没有 polyfill
, defineProperty
能支持到IE9
参考文献
https://developer.mozilla.org/zhCN/docs/Web/JavaScript/Reference/Global_Objects/Object/defineProperty