当前位置:   article > 正文

ES6中Reflect对象使用详解(附代码)_reflect.has

reflect.has

目录

目录

Reflect的方法

 Reflect.get(target, name, receiver)

Reflect.set(target, name, value, receiver)

Reflect.has(obj, name)

Reflect.deleteProperty(obj, name)

Reflect.ownKeys (target)

Reflect.preventExtensions(target)

Reflect.isExtensible (target)

Reflect.defineProperty(target, propertyKey, attributes)

Reflect.getOwnPropertyDescriptor(target, propertyKey)

Reflect.apply(func, thisArg, args)


概述

Reflect对象与Proxy对象一样,也是 ES6 为了操作对象而提供的新 API。Reflect对象的设计目的有这样几个。

1.Object对象的一些明显属于语言内部的方法(比如Object.defineProperty),放到Reflect对象上。现阶段,某些方法同时在ObjectReflect对象上部署,未来的新方法将只部署在Reflect对象上。也就是说,从Reflect对象上可以拿到语言内部的方法。

2.修改某些Object方法的返回结果,让其变得更合理。比如,Object.defineProperty(obj, name, desc)在无法定义属性时,会抛出一个错误,而Reflect.defineProperty(obj, name, desc)则会返回false

3.Object操作都变成函数行为。某些Object操作是命令式,比如name in objdelete obj[name],而Reflect.has(obj, name)Reflect.deleteProperty(obj, name)让它们变成了函数行为。

4.Reflect对象的方法与Proxy对象的方法一一对应,只要是Proxy对象的方法,就能在Reflect对象上找到对应的方法。这就让Proxy对象可以方便地调用对应的Reflect方法,完成默认行为,作为修改行为的基础。也就是说,不管Proxy怎么修改默认行为,你总可以在Reflect上获取默认行为。

Reflect的方法

  • Reflect.apply(target, thisArg, args)
  • Reflect.construct(target, args)
  • Reflect.get(target, name, receiver)
  • Reflect.set(target, name, value, receiver)
  • Reflect.defineProperty(target, name, desc)
  • Reflect.deleteProperty(target, name)
  • Reflect.has(target, name)
  • Reflect.ownKeys(target)
  • Reflect.isExtensible(target)
  • Reflect.preventExtensions(target)
  • Reflect.getOwnPropertyDescriptor(target, name)
  • Reflect.getPrototypeOf(target)
  • Reflect.setPrototypeOf(target, prototype)

 Reflect.get(target, name, receiver)

Reflect.get(target, name, receiver)

Reflect.get方法查找并返回target对象的name属性,如果没有该属性,则返回undefined。

如果name属性部署了读取函数(getter),则读取函数的this绑定receiver。

  1. <script>
  2. // Reflect.get(target, name, receiver)
  3. // Reflect.get方法查找并返回target对象的name属性,如果没有该属性,则返回undefined。
  4. let obj = {
  5. name: '再努力些吧',
  6. age: 18,
  7. work: '前端',
  8. get baz() {
  9. return this.name + '的' + this.age;
  10. },
  11. }
  12. console.log(Reflect.get(obj, 'name'));//再努力些吧
  13. console.log(Reflect.get(obj, 'age'));//18
  14. console.log(Reflect.get(obj, 'work'));//前端
  15. console.log(Reflect.get(obj, 'sex'));//undefined
  16. // 如果name属性部署了读取函数(getter),则读取函数的this绑定receiver。
  17. let newObj = {
  18. name: '不能摆烂',
  19. age: 28
  20. }
  21. console.log(Reflect.get(obj, 'baz', newObj));//不能摆烂的28
  22. </script>

Reflect.set(target, name, value, receiver)

注意:Reflect.set,方法用法复杂,我只是简单展示用法,具体详细的建议去 阮一峰的ES6 去看

Reflect.set(target, name, value, receiver)

Reflect.set方法设置target对象的name属性等于value。

如果name属性设置了赋值函数,则赋值函数的this绑定receiver。

  1. <script>
  2. // Reflect.set(target, name, value, receiver)
  3. // Reflect.set方法设置target对象的name属性等于value。
  4. let obj = {
  5. name: '再努力些吧',
  6. age: 18,
  7. work: '前端',
  8. set baz(value) {
  9. return this.age = value
  10. },
  11. }
  12. Reflect.set(obj, 'name', '今天努力了吗')
  13. console.log(obj.name);//今天努力了吗
  14. // 如果name属性设置了赋值函数,则赋值函数的this绑定receiver。
  15. let newObj = {
  16. name: '不能摆烂',
  17. age: 28
  18. }
  19. Reflect.set(obj, 'age', 15, newObj)
  20. console.log(obj.age);//18
  21. console.log(newObj.age);//15
  22. </script>

Reflect.has(obj, name)

Reflect.has方法对应name in obj里面的in运算符。

如果Reflect.has()方法的第一个参数不是对象,会报错。

  1. <script>
  2. // Reflect.has(obj, name)
  3. // Reflect.has方法对应name in obj里面的in运算符。
  4. // 如果Reflect.has()方法的第一个参数不是对象,会报错。
  5. let obj = {
  6. name: '再努力些吧',
  7. age: 18,
  8. work: '前端',
  9. }
  10. // 旧写法
  11. console.log('age' in obj);//true
  12. console.log('sex' in obj);//false
  13. // 新写法
  14. console.log(Reflect.has(obj, 'age'));//true
  15. console.log(Reflect.has(obj, 'sex'));//false
  16. </script>

Reflect.deleteProperty(obj, name)


Reflect.deleteProperty方法等同于delete obj[name],用于删除对象的属性。

该方法返回一个布尔值。如果删除成功,或者被删除的属性不存在,返回true;删除失败,被删除的属性依然存在,返回false。

如果Reflect.deleteProperty()方法的第一个参数不是对象,会报错。

  1. <script>
  2. // Reflect.deleteProperty(obj, name)
  3. // Reflect.deleteProperty方法等同于delete obj[name],用于删除对象的属性。
  4. // 该方法返回一个布尔值。如果删除成功,或者被删除的属性不存在,返回true;删除失败,被删除的属性依然存在,返回false。
  5. // 如果Reflect.deleteProperty()方法的第一个参数不是对象,会报错。
  6. let obj = {
  7. name: '再努力些吧',
  8. age: 18,
  9. work: '前端',
  10. }
  11. // 旧写法
  12. // delete obj.age
  13. // console.log(obj);//{name: '再努力些吧', work: '前端'}
  14. // 新写法
  15. console.log(Reflect.deleteProperty(obj, 'name'));//true
  16. console.log(obj);//{age: 18, work: '前端'}
  17. </script>

Reflect.ownKeys (target)

Reflect.ownKeys方法用于返回对象的所有属性,基本等同于Object.getOwnPropertyNames与Object.getOwnPropertySymbols之和

  1. <script>
  2. // Reflect.ownKeys (target)
  3. // Reflect.ownKeys方法用于返回对象的所有属性,基本等同于Object.getOwnPropertyNames与Object.getOwnPropertySymbols之和
  4. let myObject = {
  5. foo: 1,
  6. bar: 2,
  7. [Symbol.for('baz')]: 3,
  8. [Symbol.for('bing')]: 4,
  9. };
  10. //旧写法
  11. //获取对象中除了symbol之外的对象的所有属性
  12. console.log(Object.getOwnPropertyNames(myObject));//['foo', 'bar']
  13. //获取对象中symbol的对象的所有属性
  14. console.log(Object.getOwnPropertySymbols(myObject));//[Symbol(baz), Symbol(bing)]
  15. // 新写法
  16. // 获取所有属性
  17. console.log(Reflect.ownKeys(myObject));// ['foo', 'bar', Symbol(baz), Symbol(bing)]
  18. </script>

Reflect.preventExtensions(target)

Reflect.preventExtensions对应Object.preventExtensions方法,用于让一个对象变为不可扩展。它返回一个布尔值,表示是否操作成功。

  1. <script>
  2. // Reflect.preventExtensions(target)
  3. // Reflect.preventExtensions对应Object.preventExtensions方法,用于让一个对象变为不可扩展。它返回一个布尔值,表示是否操作成功。
  4. let myObject = {}
  5. // 旧写法
  6. console.log(Object.preventExtensions(myObject));//{}
  7. // 新写法
  8. console.log(Reflect.preventExtensions(myObject));//true
  9. // 如果参数不是对象,Object.preventExtensions在 ES5 环境报错,在 ES6 环境返回传入的参数,而Reflect.preventExtensions会报错。
  10. // ES5 环境
  11. // Object.preventExtensions(1) // 报错
  12. // ES6 环境
  13. // Object.preventExtensions(1) // 1
  14. // 新写法
  15. // Reflect.preventExtensions(1) // 报错
  16. </script>

Reflect.isExtensible (target)

Reflect.isExtensible方法对应Object.isExtensible,返回一个布尔值,表示当前对象是否可扩展。

如果参数不是对象,Object.isExtensible会返回false,因为非对象本来就是不可扩展的,而Reflect.isExtensible会报错。

  1. <script>
  2. // Reflect.isExtensible (target)
  3. // Reflect.isExtensible方法对应Object.isExtensible,返回一个布尔值,表示当前对象是否可扩展。
  4. let myObject = {}
  5. // 旧写法
  6. console.log(Object.isExtensible(myObject));//true
  7. // 新写法
  8. console.log(Reflect.isExtensible(myObject));//true
  9. // 如果参数不是对象,Object.isExtensible会返回false,因为非对象本来就是不可扩展的,而Reflect.isExtensible会报错。
  10. console.log(Object.isExtensible(1));// false
  11. console.log(Reflect.isExtensible(1));// 报错
  12. </script>

Reflect.defineProperty(target, propertyKey, attributes)

Reflect.defineProperty方法基本等同于Object.defineProperty,用来为对象定义属性。未来,后者会被逐渐废除,请从现在开始就使用Reflect.defineProperty代替它。

  1. <script>
  2. // Reflect.defineProperty(target, propertyKey, attributes)
  3. // Reflect.defineProperty方法基本等同于Object.defineProperty,用来为对象定义属性。未来,后者会被逐渐废除,请从现在开始就使用Reflect.defineProperty代替它。
  4. let myObject = {}
  5. // 旧写法
  6. Object.defineProperty(myObject, "name", {
  7. enumerable: false,
  8. configurable: false,
  9. writable: false,
  10. value: "再努力些把"
  11. });
  12. console.log(myObject);//{name: '再努力些把'}
  13. // 新写法
  14. Reflect.defineProperty(myObject, 'now', {
  15. value: () => Date.now()
  16. });
  17. console.log(myObject);//{name: "再努力些把" now: () => Date.now()}
  18. console.log(myObject.now());//1660548733267
  19. // 如果Reflect.defineProperty的第一个参数不是对象,就会抛出错误,比如
  20. Reflect.defineProperty(1, 'foo')。
  21. </script>

Reflect.getOwnPropertyDescriptor(target, propertyKey)

Reflect.getOwnPropertyDescriptor基本等同于Object.getOwnPropertyDescriptor,用于得到指定属性的描述对象,将来会替代掉后者。

  1. <script>
  2. // Reflect.getOwnPropertyDescriptor(target, propertyKey)
  3. // Reflect.getOwnPropertyDescriptor基本等同于Object.getOwnPropertyDescriptor,用于得到指定属性的描述对象,将来会替代掉后者。
  4. var myObject = {};
  5. Object.defineProperty(myObject, 'hidden', {
  6. value: true,
  7. enumerable: false,
  8. });
  9. Reflect.defineProperty(myObject, 'now', {
  10. value: () => Date.now()
  11. });
  12. // 旧写法
  13. var theDescriptor = Object.getOwnPropertyDescriptor(myObject, 'hidden');
  14. //{value: true, writable: false, enumerable: false, configurable: false}
  15. console.log(theDescriptor);
  16. // 新写法
  17. var theDescriptor = Reflect.getOwnPropertyDescriptor(myObject, 'now');
  18. //{writable: false, enumerable: false, configurable: false, value: ƒ}
  19. console.log(theDescriptor);
  20. // Reflect.getOwnPropertyDescriptor和Object.getOwnPropertyDescriptor的一个区别是,如果第一个参数不是对象,
  21. // Object.getOwnPropertyDescriptor(1, 'foo')不报错,返回undefined,而
  22. // Reflect.getOwnPropertyDescriptor(1, 'foo')会抛出错误,表示参数非法。
  23. </script>

Reflect.apply(func, thisArg, args)

Reflect.apply方法等同于Function.prototype.apply.call(func, thisArg, args),用于绑定this对象后执行给定函数。

一般来说,如果要绑定一个函数的this对象,可以这样写fn.apply(obj, args),但是如果函数定义了自己的apply方法,就只能写成Function.prototype.apply.call(fn, obj, args),采用Reflect对象可以简化这种操作。

  1. <script>
  2. // Reflect.apply(func, thisArg, args)
  3. // Reflect.apply方法等同于Function.prototype.apply.call(func, thisArg, args),用于绑定this对象后执行给定函数。
  4. // 一般来说,如果要绑定一个函数的this对象,可以这样写fn.apply(obj, args),但是如果函数定义了自己的apply方法,就只能写成Function.prototype.apply.call(fn, obj, args),采用Reflect对象可以简化这种操作。
  5. const ages = [11, 33, 12, 54, 18, 96];
  6. // 旧写法
  7. const youngest = Math.min.apply(Math, ages);
  8. const oldest = Math.max.apply(Math, ages);
  9. const type = Object.prototype.toString.call(youngest);
  10. // 新写法
  11. // const youngest = Reflect.apply(Math.min, Math, ages);
  12. // const oldest = Reflect.apply(Math.max, Math, ages);
  13. // const type = Reflect.apply(Object.prototype.toString, youngest, []);
  14. console.log(youngest); //11
  15. console.log(oldest); //96
  16. console.log(type); //[object Number]
  17. </script>

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/羊村懒王/article/detail/275728
推荐阅读
相关标签
  

闽ICP备14008679号