Reflect.get第三个参数recevier的作用

问题

Reflect.get三个参数target, key, receiver,前两个好理解,第三个是什么意思呢?哪些场景下会用到呢?本文尝试解释下这个问题。

解释

首先搬出MDN对应Reflect.get第三个参数recevier的解释:

Reflect(target, key, recevier)中,如果target对象中指定了getterreceiver则为getter调用时的this值。

我们先定一个原始对象obj

js 复制代码
const obj = {
    hello: 1
}

根据MDN的描述,receiver是用来指定原始对象中getterthis 。因此obj需要增加一个getter属性world,同时在getter方法使用this

js 复制代码
const obj = {
    hello: 1,
    get world() {
        return this.hello
    }
}

console.log(obj.world) // 1

这样一个符合条件的原始对象例子有了。访问obj.world输出1。

那下面的代码呢?

js 复制代码
const obj2 = {
    hello: 2
}
console.log(Reflect.get(obj, 'world', obj2)); // 2

结果返回的是2。这里解释下原因:

  1. 当前Reflect.getreceiverobj2obj2有一个属性hello为2;
  2. 根据MDN的描述,receivergetter调用的this值,因此objgetter属性的this指向obj2
  3. 当读取objworld属性时,返回this.hello,因此返回的为obj2hello值。

使用场景

通常在Proxy中我们如下返回原始对象的属性值:

js 复制代码
const pObj = new Proxy(obj, {
    get(target, key, receiver) {
        console.log('Proxy key:', key)
        return target[key]
    }
})

pObj.world; // 读取代理对象的world属性

// 输出
// Proxy key: world

因为读取了pObj的属性world,因此会进入到get方法,所以输出如上。

当然我们也可以通过Reflect.get返回属性值:

js 复制代码
const pObj = new Proxy(obj, {
    get(target, key, receiver) {
        console.log('Proxy key:', key)
        return Reflect.get(target, key, receiver)
    }
})

pObj.world; // 读取代理对象的world属性

// 输出
// Proxy key: world
// Proxy key: hello

通过Reflect.get的调用,多输出了一行hello。原因如下:

  1. 首先得知道下Proxy中的get方法中也有一个reciever,根据MDN的描述。它表示的就是当前的Proxy对象pObj
  2. 因此Reflect.get(target, key, receiver)等价于Reflect.get(target, key, pObj),读取world的时候的时候,返回this.hello,当前的this又指代的是pObj,因此读取了pObj.hello
  3. pObj.hello的读取再次进入了代理对象的get方法中,因此输出了第二行的hello

通过上面两种方式的对比,可以知道,如果使用target[key]的方法返回属性值,我们无法拦截到this.hello的读取操作;而通过Reflect.get则可以。

总结

  1. Reflect.get(target, key, receiver)方法中,receiver指代target对象中getter属性的this值。
  2. 相关代码:github.com/wdskuki/js-...
相关推荐
牧羊狼的狼6 分钟前
React 中的 HOC 和 Hooks
前端·javascript·react.js·hooks·高阶组件·hoc
知识分享小能手2 小时前
React学习教程,从入门到精通, React 属性(Props)语法知识点与案例详解(14)
前端·javascript·vue.js·学习·react.js·vue·react
魔云连洲2 小时前
深入解析:Vue与React的异步批处理更新机制
前端·vue.js·react.js
mCell2 小时前
JavaScript 的多线程能力:Worker
前端·javascript·浏览器
超级无敌攻城狮4 小时前
3 分钟学会!波浪文字动画超详细教程,从 0 到 1 实现「思考中 / 加载中」高级效果
前端
excel4 小时前
用 TensorFlow.js Node 实现猫图像识别(教学版逐步分解)
前端
gnip5 小时前
JavaScript事件流
前端·javascript
赵得C5 小时前
【前端技巧】Element Table 列标题如何优雅添加 Tooltip 提示?
前端·elementui·vue·table组件
wow_DG5 小时前
【Vue2 ✨】Vue2 入门之旅 · 进阶篇(一):响应式原理
前端·javascript·vue.js
weixin_456904275 小时前
UserManagement.vue和Profile.vue详细解释
前端·javascript·vue.js