Proxy是代理,Reflect是干嘛用的?

这篇文章发布于 2021年07月1日,星期四,23:15,归类于 JS API。 阅读 22802 次, 今日 10 次 11 条评论

 

封面图示意

一、Reflect有什么用?

一句话,Reflect没什么用,除了装装逼,让人看起来高大上以外,并不具有什么牛逼之处。

准确讲应该是这样的,Reflect更像是一种语法变体,其挂在的所有方法都能找到对应的原始语法,也就是Reflect的替代性非常强。

其实从Reflect这个单词本身字面意思就能体会出Reflect的神韵,Reflect的中文意思是“反射”,阳光照在镜子上反射,其实光子还是那些光子,只是变化了方向。

举例说明:

Reflect对象挂载了很多静态方法,所谓静态方法,就是和Math.round()这样,不需要new就可以直接使用的方法。

比较常用的两个方法就是get()set()方法:

Reflect.get(target, propertyKey[, receiver])
Reflect.set(target, propertyKey, value[, receiver])

就作用而言,等同于:

target[propertyKey]
target[propertyKey] = value;

比方说页面上有个输入框,其DOM对象变量是input,平时我们对整个输入框赋值使用的语句多半是:

input.value = 'zhangxinxu';

就可以直接使用Reflect.set()方法代替:

Reflect.set(input, 'value', 'zhangxinxu')

效果是一模一样的。

又例如,我们希望对inputvalue属性重新定义,使该输入框value属性发生变化的时候可以同时触发'change'事件,下面是使用大家普遍比较熟悉的Object.defineProperty()方法实现的示意:

const props = Object.getOwnPropertyDescriptor(input, 'value');
Object.defineProperty(input, 'value', {
    ...props,
    set (v) {
        let oldv = this.value;
        props.set.call(this, v);
        // 手动触发change事件
        if (oldv !== v) {
            this.dispatchEvent(new CustomEvent('change'));
        }
    }
});

相关介绍可以参见这篇文章:“输入框value属性赋值触发js change事件的实现

上述代码我们完全可以使用Reflect对象实现,具体的JavaScript代码如下所示。

const props = Reflect.getOwnPropertyDescriptor(HTMLInputElement.prototype, 'value');
Reflect.defineProperty(input, 'value', {
    ...props,
    set (v) {
        let oldv = this.value;
        props.set.call(this, v);
        // 手动触发change事件
        if (oldv !== v) {
            this.dispatchEvent(new CustomEvent('change'));
        }
    }
});

我们可以测试下,假设页面HTML如下:

<input id="input">

测试代码为:

input.addEventListener('change', () => {
  document.body.append('变化啦~');
});

input.value = 'zhangxinxu';

此时,就可以看到页面上出现了“变化啦~”文字,截图示意如下:

可以在这个JSBIN地址体验:https://output.jsbin.com/vugucajepa

Reflect与value赋值

二、细微差异-返回值

事物存在必有道理,如果Reflect仅仅是换了种语法,存在的意义并不大,很显然,Reflect对象的出现必然有其他的考量。

我认为其中有意义的一点就是返回值。

对于某个对象,赋值并不总是成功的。

例如,我们把 inputtype属性设置为只读,如下:

Object.defineProperty(input, 'type', {
    get () {
       return this.getAttribute('type') || 'text';
    }
});

传统的使用等于号进行的属性赋值并不能知道最后是否执行成功,需要开发者自己进行进一步的检测。

例如:

console.log(input.type = 'number');

// 输出 false
console.log(Reflect.set(input, 'type', 'number'));

上面一行赋值返回值是'number',至于改变输入框的type属性值是否成功,不得而知。

但是下面一行语句使用的Reflect.set()方法,就可以知道是否设置成功,因为Reflect.set()的返回值是true或者false(只要参数类型准确)。

除了知道执行结果外,Reflect方法还有个好处,不会因为报错而中断正常的代码逻辑执行。

例如下面的代码:

(function () {
    'use strict';

    var frozen = { 1: 81 };
    Object.freeze(frozen);

    frozen[1] = 'zhangxinxu';

    console.log('no log');
})();

会出现下面的TypeError错误:

Uncaught TypeError: Cannot assign to read only property ‘1’ of object ‘#<Object>’

后面的语句console.log('no log')就没有被执行。

但是如果使用Reflect方法,则console语句是可以执行的,例如:

(function () {
    'use strict';

    var frozen = { 1: 81 };
    Object.freeze(frozen);

    Reflect.set(frozen, '1', 'zhangxinxu');

    console.log('no log');
})();

控制台运行后的log输出值如下图所示:

log正常执行了

三、set、get方法中的receiver参数

就功能而言,Reflect.get()Reflect.set()方法和直接对象赋值没有区别,都是可以互相替代的,例如,下面两段JS效果都是一样的。

还是使用input这个DOM元素示意。

有人可能会疑问,为什么不用纯对象示意呢?

因为我发现大多数前端都对DOM不怎么感兴趣,那我就反其道行之,故意膈应人 ;另外一个原因就是DOM对象更具象,所见即所得,适合偏感性的同学的学习。

const xyInput = new Proxy(input, {
    set (target, prop, value) {
        if (prop == 'value') {
            target.dispatchEvent(new CustomEvent('change'));
        }
        target[prop] = value;

        return true;
    },
    get (target, prop) {
        return target[prop];
    }
});

input.addEventListener('change', () => {
  document.body.append('变化啦~');
});
xyInput.value = 'zhangxinxu';

和下面的JS代码效果类似的。

const xyInput = new Proxy(input, {
    set (target, prop, value) {
        if (prop == 'value') {
            target.dispatchEvent(new CustomEvent('change'));
        }
        return Reflect.set(target, prop, value);
    },
    get (target, prop) {
        return Reflect.get(target, prop);
    }
});

input.addEventListener('change', () => {
  document.body.append('变化啦~');
});
xyInput.value = 'zhangxinxu';

均有如下图所示的效果:

Reflect与value赋值

但是,当需要使用可选参数receiver参数的时候,直接对象赋值和使用Reflect赋值就会出现差异。

首先,对于DOM元素,应用receiver参数会报错。

例如下面的JS就会报错:

Reflect.set(input, 'value', 'xxx', new Proxy({}, {}));

Uncaught TypeError: Illegal invocation

但是把input换成普通的纯对象,则不会有问题,例如:

// 可以正常执行
Reflect.set({}, 'value', 'xxx', new Proxy({}, {}));

关于receiver参数

说了这么多,receiver参数到底是干嘛用的呢?

receiver是接受者的意思,表示调用对应属性或方法的主体对象,通常情况下,receiver参数是无需使用的,但是如果发生了继承,为了明确调用主体,receiver参数就需要出马了。

比方说下面这个例子:

let miaoMiao = {
  _name: '疫苗',
  get name () {
    return this._name;
  }
}
let miaoXy = new Proxy(miaoMiao, {
  get (target, prop, receiver) {
    return target[prop];
  }
});

let kexingMiao = {
  __proto__: miaoXy,
  _name: '科兴疫苗'
};

// 结果是疫苗
console.log(kexingMiao.name);

实际上,这里预期显示应该是“科兴疫苗”,而不是“疫苗”。

这个时候,就需要使用receiver参数了,代码变化部分参见下面标红的那一行:

let miaoMiao = {
  _name: '疫苗',
  get name () {
    return this._name;
  }
}
let miaoXy = new Proxy(miaoMiao, {
  get (target, prop, receiver) {
    return Reflect.get(target, prop, receiver);
    // 也可以简写为 Reflect.get(...arguments) 
  }
});

let kexingMiao = {
  __proto__: miaoXy,
  _name: '科兴疫苗'
};

// 结果是科兴疫苗 
console.log(kexingMiao.name);

此时,运行结果就是预期的“科兴疫苗”了,如下截图所示:

科兴疫苗

这就是receiver参数的作用,可以把调用对象当作target参数,而不是原始Proxy构造的对象。

四、其他以及结束语

Reflect对象经常和Proxy代理一起使用,原因有三点:

  1. Reflect提供的所有静态方法和Proxy第2个handle参数方法是一模一样的。具体见后面的对比描述。
  2. Proxy get/set()方法需要的返回值正是Reflect的get/set方法的返回值,可以天然配合使用,比直接对象赋值/获取值要更方便和准确。
  3. receiver参数具有不可替代性。

下表是自己整理的Reflect静态方法和对应的其他函数或功能符。

Reflect方法 类似于
Reflect.apply(target, thisArgument, argumentsList) Function.prototype.apply()
Reflect.construct(target, argumentsList[, newTarget]) new target(…args)
Reflect.defineProperty(target, prop, attributes) Object.defineProperty()
Reflect.deleteProperty(target, prop) delete target[name]
Reflect.get(target, prop[, receiver]) target[name]
Reflect.getOwnPropertyDescriptor(target, prop) Object.getOwnPropertyDescriptor()
Reflect.getPrototypeOf(target) Object.getPrototypeOf()
Reflect.has(target, prop) in 运算符
Reflect.isExtensible(target) Object.isExtensible()
Reflect.ownKeys(target) Object.keys()
Reflect.preventExtensions(target) Object.preventExtensions()
Reflect.set(target, prop, value[, receiver]) target[prop] = value
Reflect.setPrototypeOf(target, prototype) Object.setPrototypeOf()

正是人如其名,Reflect就是其他方法、操作符的“反射”。

好,以上就是本文的内容,带大家了解了下Reflect的七七八八。

希望可以对大家的学习有所帮助。

欢迎转发,欢迎分享,谢谢谢谢!

参考文档

(本篇完)

分享到:


发表评论(目前11 条评论)

  1. 落叶也疯狂说道:

  2. lei说道:

    – 为啥加了一句 console 就会变成死循环
    “`
    let miaoMiao = {
    _name: ‘疫苗’,
    get name () {
    return this._name;
    }
    }
    let miaoXy = new Proxy(miaoMiao, {
    get (target, prop, receiver) {
    // 加了这一句 #######
    console.log(receiver)
    return Reflect.get(target, prop, receiver);
    // 也可以简写为 Reflect.get(…arguments)
    }
    });

    let kexingMiao = {
    __proto__: miaoXy,
    _name: ‘科兴疫苗’
    };

    // 结果是科兴疫苗
    console.log(kexingMiao.name);
    “`

  3. cyh41说道:

    还有个问题就是,在proxy里set当前prop时,再set其他prop,如果直接target[‘其它prop’] = ‘xxx’ proxy不回拦截,得receiver[‘其它prop’] =’xxx’ 才会拦截

  4. zhh说道:

    回复juice_157
    跟console.log实现有关,有两个原因影响
    1. console.log时会调用沿着原型链查找是否有splice方法来判断是不是数组,当查找到proxy时候,被劫持又遇到console.log(receiver),就会陷入死循环。
    2. chrome浏览器环境下console.log实现时会调用toString方法,会沿着原型链搜索[Symbol.toStringTag]方法,跟上个原因一样,找到proxy时会进入死循环。https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Symbol/toStringTag

    chrome浏览器下验证测试的话可以把代码改成:
    let kexingMiao = {
    __proto__: miaoXy,
    _name: “科兴疫苗”,
    splice: undefined,
    [Symbol.toStringTag]() {
    return ‘Validator’;
    }
    };
    这样在当前对象找到那两个方法后,就不会沿着原型链继续搜索,就不会陷入死循环。

  5. will说道:

    就是把一系列元编程的工具放到一个命名空间下,之前的太零散了,所以很多方法就跟Object下的重复了。

  6. 坐地吸土说道:

    再捞一捞reflect

  7. juice_157说道:

    receiver那块还是不太明白,我在return Reflect.get(target, prop, receiver);之前加一个console.log(receiver),就陷入了一个死循环

    • cs404说道:

      能提供一下具体代码吗,我这里尝试是不会死循环的
      “`js
      const target = {};
      const obj = new Proxy(target, {
      get(target, key, receiver) {
      console.log(‘遍历中…’)
      console.log(receiver)// Proxy {}
      return Reflect.get(target, key, receiver)
      }
      })
      obj.text
      “`

  8. 崮生说道:

    就是绕过 proxy 的方法

  9. Vvvvv说道:

    学会了好像又没学会…