vue3中的Proxy为什么一定要用Reflect

广告:宝塔Linux面板高效运维的服务器管理软件 点击【 https://www.bt.cn/p/uNLv1L 】立即购买

vue3中的Proxy为什么一定要用Reflect

用过vue的知道,vue的响应实现用的Proxy,且里面是配合Reflect用的,查看Proxy和Reflect文档最显眼的是Reflect对象的静态方法和Proxy代理方法的命名相同,Reflect可以操作对象使用, proxy可以代理对象,但没有找到为啥有时一定要在Proxy代理方法中使用Reflect

基本操作

Reflect对象的静态方法和Proxy代理方法的命名相同,都有13种,示例get,set如下

const tempObj = { a: 1 };Reflect.get(tempObj, 'a'); // 返回 1Reflect.set(tempObj, 'a', 2); // 返回true 表示设置成功, a的值变2const tempObj1 = { a: 1 };const handler = {  get: function (obj, prop, receiver) {    return prop === 'a' ? 1000 : obj[prop];  },  set: function (obj, prop, value, receiver) {    console.log(prop);    obj[prop] = prop === 'a' ? 6 : value;    return true;  },};const proxyObj = new Proxy(tempObj1, handler);proxyObj.a; // proxyObj => {a: 1000}proxyObj.a = 2; // proxyObj => {a: 6}
登录后复制疑问

如果Proxy不做其它操作直接正常返回

const tempObj1 = { a: 1 };const handler = {  get: function (obj, prop, receiver) {    return obj[prop];  },  set: function (obj, prop, value, receiver) {    obj[prop] = value    return true;  },};const proxyObj = new Proxy(tempObj1, handler);proxyObj.a; // proxyObj => {a: 1}proxyObj.a = 2; // proxyObj => {a: 2}
登录后复制

以上面情况完Proxy可以不使用Reflect处理拦截,比使用Reflect简单多了

不一样的对象, 带有get的对象

const tempObj1 = {  a: 1,  get value() {    console.log(this === proxyObj); // false    return this.a;  },};const handler = {  get: function (obj, prop, receiver) {    return obj[prop];  },  set: function (obj, prop, value, receiver) {    obj[prop] = value;    return true;  },};const proxyObj = new Proxy(tempObj1, handler);proxyObj.value; // 1
登录后复制

上面value中的打印的值为false,期望的结果应该true, 但应该代理中用的原对象取值所以this指向了原对象,所以值为false

虽然this指错了,但得到值还是正确定,这不是一定的理由

const parent = {  a: 1,  get value() {    console.log(this === child); // false    return this.a;  },};const handler = {  get: function (obj, prop, receiver) {    return obj[prop];  },  set: function (obj, prop, value, receiver) {    obj[prop] = value;    return true;  },};const proxyObj = new Proxy(parent, handler);const child = Object.setPrototypeOf({ a: 2 }, proxyObj);child.value; // 1
登录后复制

这就有问题了,输出的结果都和期望的不一样了,this应该指向child,但指向了parent

Reflect上场

要是Reflect.get(obj, prop)换成obj[prop],这等于没换,意义和结果是一样的,这不是还有一个receiver参数没有用嘛

const parent = {  a: 1,  get value() {    console.log(this === child); // true    return this.a;  },};const handler = {  get: function (obj, prop, receiver) {    Reflect.get(obj, prop)   - return obj[prop];   + retrun Reflect.get(obj, prop, receiver)  },  set: function (obj, prop, value, receiver) {   - obj[prop] = value;   + Reflect.get(obj, prop, value, receiver)    return true;  },};const proxyObj = new Proxy(parent, handler);const child = Object.setPrototypeOf({ a: 2 }, proxyObj);child.value; // 2
登录后复制

this指向正确,结果也当然和期望一致,receiver的不是指代理对象,也不是指原对象,而是执行上下文(有句话是这么说的,不用特定方式改变this的情况下,谁调用指向谁,这就是期望的),这里child调用的value所以期望的指向应该是child, 这里你可能想到直接用receiver[prop]不行了,这样会出现执行溢出,receiver[prop]相当于child.valuechild.value还没执行完,receiver[prop]又执行了,就会无限在执行

Reflect.get(target, key, receiver)中的receiver参数修改了this指向,不加this指向target, 加了后指向receiver

代理对象中有用到this时一定要用到Reflect,这样才能得到一直符合期望的值

总结

vue3中的代理对象到得的都是符合期望的值,在拦截中做了收集和更新,所以一定要在Proxy的拦截函数中使用Reflect处理

怎么一直拿到符合期望的值,代理了像没有代理一样
  get: function (...arg) {    return Reflect.get(...arg);  },
登录后复制

以上就是vue3中的Proxy为什么一定要用Reflect的详细内容,更多请关注9543建站博客其它相关文章!

广告:SSL证书一年128.66元起,点击购买~~~

9543建站博客
一个专注于网站开发、微信开发的技术类纯净博客。
作者头像
admin创始人

肥猫,知名SEO博客站长,14年SEO经验。

上一篇:node.js gm是什么
下一篇:laravel 循环 错误

发表评论

关闭广告
关闭广告