当前位置:网站首页>JS高级 之 Proxy-Reflect 使用详解

JS高级 之 Proxy-Reflect 使用详解

2022-08-03 02:06:00 玄鱼殇

目录

一、以前监听对象 Object.defineProperty() - JavaScript | MDN

1. 使用Object.defineProperty

2. 缺点 

二、Proxy Proxy - JavaScript | MDN

1. 基本使用

2. set和get捕获器

2. Proxy所有捕获器

三、Reflect Reflect - JavaScript | MDN

1. 作用

2. 出现的理由 ​​​​​​

3. 和Object的区别

4.  Reflect的常见方法

5. Reflect和Proxy搭配来使用

6.  Reflect和Proxy搭配来使用


一、以前监听对象 Object.defineProperty() - JavaScript | MDN

1. 使用Object.defineProperty

const obj = {
  name: 'star',
  age: 14
};
for (const [key, value] of Object.entries(obj)) {
  let _value = value;
  // 监听
  Object.defineProperty(obj, key, {
    enumerable: true,
    set(value) {
      console.log('set:', value);
      // 通过闭包使用外部的值,   obj.name = value =>  这样会死循环,注意
      _value = value;
    },
    get() {
      console.log('get:', _value);
      // 通过闭包使用外部的值,   obj.name = value =>  这样会死循环,注意
      return _value;
    }
  });
}
// get监听到
console.log(obj.name);
// set监听到
obj.name = 'coder';
// get监听到
console.log(obj.name);

2. 缺点 

1. Object.defineProperty设计的初衷,不是为了去监听截止一个对象中所有的属性的

2. 想监听更加丰富的操作,比如新增属性、删除属性,那么Object.defineProperty是无能为力的

二、Proxy Proxy - JavaScript | MDN

 在ES6中,新增了一个Proxy类,是用于创建一个代理的

  • 也就是说,如果希望监听一个对象的相关操作,可以先创建一个代理对象(Proxy对象)
  • 之后对该对象的所有操作,都通过代理对象来完成,代理对象可以监听想要对原对象进行哪些操作
  • 想要侦听某些具体的操作,那么就可以在handler中添加对应的捕捉器(Trap)

1. 基本使用

const obj = {
  name: 'star',
  age: 18,
  height: 1.88
};

const objProxy = new Proxy(obj, {});

2. set和get捕获器

set函数有四个参数:

  • target:目标对象(侦听的对象)
  • property:将被设置的属性key
  • value:新属性值
  • receiver:调用的代理对象

get函数有三个参数:

  • target:目标对象(侦听的对象)
  • property:被获取的属性key
  • receiver:调用的代理对象
const obj = {
  name: 'star',
  age: 18,
  height: 1.88
};
// 1. 创建proxy代理对象
const objProxy = new Proxy(obj, {
  // 2. 设置set捕获器
  set(target, key, newValue) {
    console.log(`监听到${key}设置值`);
    target[key] = newValue;
  },
  // 3. 设置get捕获器
  get(target, key) {
    console.log(`监听到${key}获取值`);
    return target[key];
  }
});
// 触发get捕获器
console.log(objProxy.name);
// 触发set捕获器
objProxy.name = 'coder';
// 触发get捕获器
console.log(objProxy.name);

// 4. 新增属性也能被监听到
objProxy.address = '广州市';
// 触发set捕获器
console.log(objProxy.address);
// 添加成功
console.log(obj); // {name: 'coder', age: 18, height: 1.88, address: '广州市'}

2. Proxy所有捕获器

  • handler.getPrototypeOf()
    • Object.getPrototypeOf 方法的捕捉器,获取对象的原型
  • handler.setPrototypeOf()
    • Object.setPrototypeOf 方法的捕捉器,设置对象的原型
  • handler.isExtensible()
    • Object.isExtensible 方法的捕捉器,判断是否可以新增属性
  • handler.preventExtensions()
    • Object.preventExtensions 方法的捕捉器,阻止对象扩展
  • handler.getOwnPropertyDescriptor()
    • Object.getOwnPropertyDescriptor 方法的捕捉器,获取自己的属性描述符
  • handler.defineProperty()
    • Object.defineProperty 方法的捕捉器,定义自己的属性描述符
  • handler.ownKeys()
    • Object.getOwnPropertyNames 方法和Object.getOwnPropertySymbols 方法的捕捉器
  • handler.has()
    • in 操作符的捕捉器, 用in操作符判断的时候
  • handler.get()
    • 属性读取操作的捕捉器
  • handler.set()
    • 属性设置操作的捕捉器
  • handler.deleteProperty()
    • delete 操作符的捕捉器
  • handler.apply()
    • 函数调用操作的捕捉器,函数调用apply时触发
  • handler.construct()
    • new 操作符的捕捉器,函数被new时触发

三、Reflect Reflect - JavaScript | MDN

Reflect也是ES6新增的一个API,是一个对象,意思是反射

1. 作用

  • 主要提供了很多操作JavaScript对象的方法,有点像Object中操作对象的方法
  • Reflect.getPrototypeOf(target) 类似于 Object.getPrototypeOf()
  • Reflect.defineProperty(target, propertyKey, attributes) 类似于Object.defineProperty()

2. 出现的理由 ​​​​​​

比较 Reflect 和 Object 方法 - JavaScript | MDN

  1. 是因为在早期的ECMA规范中没有考虑到这种对 对象本身 的操作如何设计会更加规范,所以将这些API放到了Object上面
  2. 但是Object作为一个构造函数,这些操作实际上放到它身上并不合适
  3. 同时Object作为所有类的父类,导致Object非常的臃肿
  4. 另外还包含一些类似in、delete的操作符,有点奇怪
  5. ----------------------------------------------
  6. 所以ES6新增了Reflect,让对对象的操作全部集中到了Reflect对象上
  7. Reflect同时配合Proxy,可以做到不操作原对象

看文档~~~非常nice的东西

3. 和Object的区别

大部分操作完对象后都会返回bool值,用来告知是否操作成功

// 开启严格模式
'use strict';
const obj = {
  name: 'why',
  age: 18
};
Object.defineProperty(obj, 'name', {
  configurable: false
});

// 用以前的方式进行操作
// 1. 不确定能否删除,如果这个对象定义过属性描述符configurable为false,那么就不能删除
// 严格模式下还会报错
delete obj.name;
// 2. 判断是否删除成功,需要这样判断,这样判断还不严谨,如果对象的隐式原型上或着之上有这个属性,那么就判断不成功
if (obj.name) {
  console.log('name没有删除成功');
} else {
  console.log('name删除成功');
}

// 使用Reflect
// 1. 就算配置了不可删除,但是使用这个deleteProperty不会报错而导致卡在这
// 2. 删除成功或者失败会返回bool值,代码更加严谨
if (Reflect.deleteProperty(obj, 'name')) {
  console.log('name删除成功');
} else {
  console.log('name没有删除成功');
}

console.log(obj);

4.  Reflect的常见方法

是和Proxy是一一对应的,也是13个

  • Reflect.getPrototypeOf(target)
    • 类似于 Object.getPrototypeOf()
  • Reflect.setPrototypeOf(target, prototype)
    • 设置对象原型的函数. 返回一个 Boolean, 如果更新成功,则返回
      true
  • Reflect.isExtensible(target)
    • 类似于 Object.isExtensible()
  • Reflect.preventExtensions(target)
    • 类似于 Object.preventExtensions()。返回一个Boolean
  • Reflect.getOwnPropertyDescriptor(target, propertyKey)
    • 类似于 Object.getOwnPropertyDescriptor()。如果对象中存在该属性,则返回对应的属性描述符, 否则返回 undefined
  • Reflect.defineProperty(target, propertyKey, attributes)
    • 和 Object.defineProperty() 类似。如果设置成功就会返回 true
  • Reflect.ownKeys(target)
    • 返回一个包含所有自身属性(不包含继承属性)的数组。(类似于Object.keys(), 但不会受enumerable影响)
  • Reflect.has(target, propertyKey)
    • 判断一个对象是否存在某个属性,和 in 运算符 的功能完全相同
  • Reflect.get(target, propertyKey[, receiver])
    • 获取对象身上某个属性的值,类似于 target[name]
  • Reflect.set(target, propertyKey, value[, receiver])
    • 将值分配给属性的函数。返回一个Boolean,如果更新成功,则返回true
  • Reflect.deleteProperty(target, propertyKey)
    • 作为函数的delete操作符,相当于执行 delete target[name]
  • Reflect.apply(target, thisArgument, argumentsList)
    • 对一个函数进行调用操作,同时可以传入一个数组作为调用参数。和
      Function.prototype.apply() 功能类似
  • Reflect.construct(target, argumentsList[, newTarget])
    • 对构造函数进行 new 操作,相当于执行 new target(...args)

5. Reflect和Proxy搭配来使用

主要目的 :

  • 代码更规范、不直接操作原对象
  • 用Proxy监听、用Reflect执行
  • 执行完后还知道是否操作成功
const obj = {
  name: 'why',
  age: 18
};

const objProxy = new Proxy(obj, {
  // 监听设置
  set(target, key, newValue) {
    // 以前写法,不好,不知是否操作成功
    // target[key] = newValue
    // 现在,nice!
    const isSuccess = Reflect.set(target, key, newValue);
    if (!isSuccess) {
      throw new Error(`set ${key} failure`);
    }
  },
  // 监听获取
  get(target, key) {
    // 不直接获取,使用Reflect
    return Reflect.get(target, key);
  },
  // 监听判断
  has(target, key) {
    console.log('has');
    // 不直接判断,使用Reflect
    return Reflect.has(target, key);
  }
});

// 操作代理对象
objProxy.name = 'kobe'; // 触发set操作
console.log('name' in objProxy); // 触发set操作 true
console.log(obj); // {name: 'kobe', age: 18}

6.  Reflect和Proxy搭配来使用

// 父类
function Person(name, age) {
  this.name = name;
  this.age = age;
}
// 子类
function Student(name, age) {
  // 可以不使用构造函数继承   在babel中的源码中就做了类似的判断
  // Person.call(this, name, age)
  // 创建一个对象,该对象的隐式原型指向Student,同时使用Person的构造方法
  const _this = Reflect.construct(Person, [name, age], Student);
  return _this;
}

// const stu = new Student("why", 18)
const stu = new Student('coder', 18);
console.log(stu); // Student {name: 'coder', age: 18}
console.log(stu.__proto__ === Student.prototype); // true

原网站

版权声明
本文为[玄鱼殇]所创,转载请带上原文链接,感谢
https://blog.csdn.net/a15297701931/article/details/126107975