Vue响应式数据之Object
在阅读《深入浅出 Vue.js》与《Vue.js 设计与实现》,了解到 vue 是如何侦测数据,同时自己在接触 js 逆向时也常常会用到。于是就准备写篇 js 如何监听数据变化,这篇为监听 Object 数据。
Object.defineproperty
const data = {
username: 'kuizuo',
password: 'a123456',
}
function defineReactive(data, key, val) {
Object.defineProperty(data, key, {
enumerable: true,
configurable: true,
get() {
console.log('GET', val)
return val
},
set(newVal) {
if (val === newVal) return
val = newVal
console.log('SET', val)
},
})
}
function observe(data) {
Object.keys(data).forEach(function (key) {
defineReactive(data, key, data[key])
})
}
observe(data)
data.username
data.username = 'MongoRolls'
从上面的代码中就可以发现,只要取值与赋值就会进入 get 和 set 函数内,在这里面便可以实现一些功能,例如 Vue 中收集依赖,在想监听浏览器中 cookies 的取值与赋值,就可以使用如下代码
!(function () {
let cookie = document.cookie
Object.defineProperty(document, 'cookie', {
get() {
console.log('cookie get', cookie)
return cookie
},
set(newVal) {
cookie = newVal
console.log('cookie set', cookie)
},
})
})()
使用 object.defineproperty 能监听对象上的某个属性修改与获取,但是无法监听到对象属性的增和删。这在 es5 是无法实现的,因为还不支持元编程。这也就是为什么 Vue2 中对于对象无法监听到 data 的某个属性增加与删除了
var vm = new Vue({
data: {
a: 1,
},
})
// `vm.a` 是响应式的
vm.b = 2
// `vm.b` 是非响应式的
Proxy 与 Reflect
但在 ES6 中提供了 Proxy 可以实现元编程,同时 Vue3 也使用 Proxy 来重写响应式系统。所以就很有必要去了解该 API
function reactive(target) {
return new Proxy(target, {
get(target, key) {
const res = target[key]
console.log('GET', key, res)
return res
},
set(target, key, newValue) {
target[key] = newValue
console.log('SET', key, newValue)
},
deleteProperty(target, key) {
console.log('DELETE', key)
delete target[key]
},
})
}
但上述写法中使用了target[key]
是能获取到 target 的值,但可能会存在一定隐患(如 this 问题),所以更推荐使用Reflect
对象的方法,如下
function reactive(target) {
return new Proxy(target, {
get(target, key, receiver) {
const res = Reflect.get(target, key, receiver)
console.log('GET', key, res)
return res
},
set(target, key, newValue, receiver) {
const res = Reflect.set(target, key, newValue, receiver)
console.log('SET', key, newValue)
return res
},
deleteProperty(target, key) {
const res = Reflect.deleteProperty(target, key)
console.log('DELETE', key)
return res
},
})
}
调用如下
const target = {
foo: 1,
bar: 1,
}
let p = reactive(target)
p.foo++
delete p.bar
console.log(target)
输出内容如下
GET foo 1
SET foo 2
DELETE bar
{ foo: 2 }
其中这里的 get,set,deleteProperty 可以拦截到对象属性的取值,赋值与删除的操作。相比 Object.defineproperty 除了好用外,可操作空间也大。
this 问题
如果 target 对象存在 this,那么不做任何拦截的情况下,target 的 this 所指向的是 target,而不是代理对象 proxy
const target = {
m: function () {
console.log(this === proxy)
},
}
const handler = {}
const proxy = new Proxy(target, handler)
target.m() // false
proxy.m() // true
具体可看:this 问题