vue3出来也有段时间了,无疑被说的最多应该就是vue的数据拦截用proxy重写了(以前用的是Object.defineProperty)和 Composition API 了。本篇文,来体验一下proxy的拦截能力到底有多优秀。javascript
这里经过vue1.x,vue2.x
时的数据拦截来讲一下Object.defineProperty
。前端
提早说明,这里的数据拦截,不包含vue依赖收集,视图更新,否则又给可爱的掘友骂死。vue
先来用Object.defineProperty
实现一下对象的拦截。java
let data = {
m:234,
n:[1,34,4,5676],
h:{
c:34
}
}
function observer(data){
if(typeof data === 'object'){
Object.keys(data).forEach(key=>{
defineReactive(data,key,data[key])
})
}
}
function defineReactive(obj,key,val){
Object.defineProperty(obj,key,{
get(){
console.log('get')
return val
},
set(newVal){
console.log('set')
if(newVal !== val ) val = newVal
}
})
}
复制代码
上面经过遍历data
的数据,进行了一次简单的拦截;看似没有问题,但若是咱们改变data.h.c
是不会触发set钩子的
,为何?由于这里尚未实现递归
,因此只拦截了最表面的一层
,里面的则没有被拦截。数组
递归拦截对象markdown
function defineReactive(obj,key,val){
observer(val)
Object.defineProperty(obj,key,{
get(){
console.log('get')
return val
},
set(newVal){
console.log('set')
if(newVal !== val ) val = newVal
}
})
}
复制代码
递归拦截,只要在defineReactive函数
再调一次observer函数
把要拦截的值传给它就行。这样,就实现了对象的多层拦截。可是呢,如今是拦截不到数组的,当咱们调用push,pop等方法
它是不会触发set钩子的
,为何?由于Object.defineProperty
压根就不支持数组的拦截。既然它不支持,那么咱们只能拦截它的这些('push', 'pop', 'shift', 'unshift', 'splice', 'sort', 'reverse')
改变自身数据的方法了。app
function arrayMethods(){
const arrProto = Array.prototype
const arrayMethods = Object.create(arrProto)
const methods = ['push', 'pop', 'shift', 'unshift', 'splice', 'sort', 'reverse']
methods.forEach(function (method) {
const original = arrProto[method]
Object.defineProperty(arrayMethods, method, {
value: function v(...args) {
console.log('set arrayMethods')
return original.apply(this, args)
}
})
})
return arrayMethods
}
复制代码
以上就是对这些数组的原型方法进行了一个拦截,而后把它覆盖要拦截的数组的原型就行,下面简单修改一下observer函数
function observer(data){
if(typeof data === 'object'){
if(Array.isArray(data)){
data.__proto__ = arrayMethods()
}else{
Object.keys(data).forEach(key=>{
defineReactive(data,key,data[key])
})
}
}
}
复制代码
在vue
中,还会判断该key
有没有__proto__
,若是没有就直接把这些方法放到这个key的自身上,若是有就直接覆盖这个__proto__
。post
function arrayMethods(){
const arrProto = Array.prototype
const arrayMethods = Object.create(arrProto)
const methods = ['push', 'pop', 'shift', 'unshift', 'splice', 'sort', 'reverse']
methods.forEach(function (method) {
const original = arrProto[method]
Object.defineProperty(arrayMethods, method, {
value: function v(...args) {
console.log('set arrayMethods')
return original.apply(this, args)
}
})
})
return arrayMethods
}
function observer(data){
if(typeof data === 'object'){
if(Array.isArray(data)){
data.__proto__ = arrayMethods()
}else{
Object.keys(data).forEach(key=>{
defineReactive(data,key,data[key])
})
}
}
}
function defineReactive(obj,key,val){
observer(val)
Object.defineProperty(obj,key,{
get(){
console.log('get')
return val
},
set(newVal){
console.log('set')
if(newVal !== val ) val = newVal
}
})
}
observer(data)
复制代码
以上,就完成了对对象和数组
的拦截(说明:vue2.x中的实现会比这里复杂,但大概思路和大概实现是这样)
,看似辛苦点,换来一个完美的结果挺不错的。但真的是你想的那样吗?试一下调用data.n[1] = xxx
,它是不会触发set钩子的
,这也是在proxy
出现以前,无能无力的,因此在vue中提供了$set,$delete
API。ui
这里就不介绍proxy了,就当你对它有了解过了。直接上代码
let data = {
m:234,
n:[1,34,4,5676],
h:{
c:34
}
}
function defineReactive(obj){
Object.keys(obj).forEach((key) => {
if(typeof obj[key] === 'object'){
obj[key] = defineReactive(obj[key])
}
})
return new Proxy(obj,{
get(target,key){
console.log('get')
return target[key]
},
set(target,key,val){
console.log('set')
return target[key] = val
}
})
}
data = defineReactive(data)
复制代码
就这么一点代码就实现了对对象和数组
的拦截(说明:这里不是vue3的实现方法,vue3怎么实现的,我还不知道,还没看过它的源码,有兴趣本身去看一下,而后顺便告诉我一下怎么实现的,哈哈哈)
,Object.defineProperty
实现不了的,它能实现;Object.defineProperty
实现的了,它也能实现。不管你调push,pop等方法
它能拦截,你调data.n[1] = xxx
也能拦截,简直不要太爽,这个两个版本的实现,给我我的的感受就是一个韩红版(肉多腿短),一个迪丽热巴版(苗条腿长),哈哈哈,本身品。
这里只是经过对对象和数组
的拦截,来体验了一下proxy的威力;proxy能作的远远不止这样。
了解proxy更多特性,能够看看我以前整理比较基础《Proxy入门》
了解proxy更多用法,能够看看,前端小智的《Proxy 的巧用》
若是你想了解更多proxy的特性,和更多的用例能够,看看以上两篇文章(自愿看哈,我不想给骂,说什么广告,瞎jb推荐之类的,🐶保命)。