Proxy-Reflect

简介: JS查漏补缺系列是我在学习JS高级语法时做的笔记,通过实践费曼学习法进一步加深自己对其的理解,也希望别人能通过我的笔记能学习到相关的知识点。这一次我们来了解Proxy-Reflect

监听对象的操作

场景:监听一个对象中的属性被设置或获取的过程

方式一:可以利用 Object.defineProperty实现
const obj = {
  name: "唔西迪西",
  age: 18
}

Object.keys(obj).forEach(key => {
  let value = obj[key]

  Object.defineProperty(obj, key, {
    get: function() {
      console.log(`监听到obj对象的${key}属性被访问了`)
      return value
    },
    set: function(newValue) {
      console.log(`监听到obj对象的${key}属性被设置值`)
      value = newValue
    }
  })
})

obj.name = "玛卡巴卡"
obj.age = 30

console.log(obj.name)
console.log(obj.age)

缺点:

  1. Object.defineProperty 就不是用来监听对象的属性的,但利用它的特性可以实现
  2. Object.defineProperty 不能监听新增属性、删除属性的操作等其它操作
  3. 无法监听数组变化,Vue 通过 Hack 改写八种数组方法实现

Proxy

ES6新增的一个类 , Proxy 对象用于创建 一个对象的代理,从而实现基本操作的拦截和自定义(如属性查找、赋值、枚举、函数调用等)。
所以实现监听时监听的不是原来的对象,而是监听原来对象的代理对象

使用:

// target:要代理的对象;   handler:里面包含捕获器方法
const proxy = new Proxy(target, handler);

MDN对Proxy的传入参数解释:

  • handler:包含捕捉器(Trap)的占位符对象,可译为处理器对象
  • traps:提供属性访问的方法,这类似于操作系统中捕获器的概念
  • target:被 Proxy 处理虚拟化的对象,它常被作为代理的存储后端,根据目标验证关于对象不可扩展性或不可配置属性的不变量(保持不变的语义)

现在我们通过Proxy中捕获器的重写来实现监听一个对象中的属性被设置或获取的过程:

const obj = {
  name: "唔西迪西",
  age: 18
}

const proxy = new Proxy(obj, {
  // 获取值时的捕获器
  get: function(target, key) {
    console.log(`监听到对象的${key}属性被访问了`, target)
    return target[key]
  },

  // 设置值时的捕获器
  set: function(target, key, newValue,receiver) {
    console.log(`监听到对象的${key}属性被设置值`, target)
    target[key] = newValue
  }
  })
});

proxy.name = "玛卡巴卡"
proxy.age = 30

console.log(proxy.name)
console.log(proxy.age)

Proxy捕获器

上面已经使用了get和set捕获器,接下来介绍一些其他常见的捕获器
  1. in 操作符的捕捉器

handler.has()方法是针对in操作符的代理方法

const obj = {
  name: "唔西迪西",
  age: 18
}

const objProxy = new Proxy(obj, {
  // 监听in的捕获器
  // has 有target, key,没有 receiver
  has: function(target, key) {
    console.log(`监听到对象的${key}属性in操作`, target)
    return key in target
  }
})

// in操作符
console.log("name" in objProxy)
  1. delete 操作符的捕捉器。

handler.deleteProperty() 方法用于拦截对对象属性的 delete 操作。

const obj = {
  name: "唔西迪西",
  age: 18
}

const objProxy = new Proxy(obj, {
  // 监听delete的捕获器
  deleteProperty: function(target, key) {
    console.log(`监听到对象的${key}属性in操作`, target)
    delete target[key]
  }
})
// delete操作
delete objProxy.name

Reflect

ES6新增的一个API,它是一个内置的对象,它提供拦截 JavaScript 操作的方法。但它不是一个函数对象,因此它是不可构造的。
一般我们见到Reflect是跟Proxy一起使用的

Reflect 对象是一个内置对象,提供了与 JavaScript 对象交互的方法,与原来我们学过的Object方法类似,但还是有一些差异,可以看看MDN对它们差异的总结:https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Reflect/Comparing_Reflect_and_Object_methods

现在我们通过Proxy和Reflect来实现开头的场景:

const obj = {
  name: "唔西迪西",
  age: 18
}

const proxy = new Proxy(obj, {
  get: function(target, key) {
    console.log(`监听到对象的${key}属性被访问了`, target)
    return Reflect.get(target, key)
  },

  set: function(target, key, newValue) {
    console.log(`监听到对象的${key}属性被设置值`, target)
    Reflect.set(target, key, newValue) // 返回的是Boolean
  }
  })
});

proxy.name = "玛卡巴卡"
console.log(proxy.name)

补充:只有get和set的参数有receiver,receiver是指创建出来的代理对象

get: function(target, key,receiver) {}
set: function(target, key, newValue,receiver) {}
目录
相关文章
|
iOS开发 MacOS Python
Mac:安装Python3并配置环境变量(本地多个Python版本选择配置)
Mac:安装Python3并配置环境变量(本地多个Python版本选择配置)
17118 0
Mac:安装Python3并配置环境变量(本地多个Python版本选择配置)
|
前端开发 JavaScript
前端祖传三件套JavaScript的ES6+之Proxy/Reflect
在现代前端开发中,JavaScript已经成为了不可或缺的语言。作为前端祖传三件套之一,它一直都在不断发展和进化。其中,ES6+是最受欢迎的版本之一,因为它引入了许多新功能和特性,能够更加方便有效地编写代码。本文将介绍其中之一的Proxy/Reflect。
241 0
|
5天前
|
数据采集 人工智能 安全
|
15天前
|
云安全 监控 安全
|
1天前
|
存储 SQL 大数据
删库跑路?别慌!Time Travel 带你穿回昨天的数据世界
删库跑路?别慌!Time Travel 带你穿回昨天的数据世界
234 156
|
8天前
|
SQL 自然语言处理 调度
Agent Skills 的一次工程实践
**本文采用 Agent Skills 实现整体智能体**,开发框架采用 AgentScope,模型使用 **qwen3-max**。Agent Skills 是 Anthropic 新推出的一种有别于mcp server的一种开发方式,用于为 AI **引入可共享的专业技能**。经验封装到**可发现、可复用的能力单元**中,每个技能以文件夹形式存在,包含特定任务的指导性说明(SKILL.md 文件)、脚本代码和资源等 。大模型可以根据需要动态加载这些技能,从而扩展自身的功能。目前不少国内外的一些框架也开始支持此种的开发方式,详细介绍如下。
595 5
|
11天前
|
人工智能 自然语言处理 API
一句话生成拓扑图!AI+Draw.io 封神开源组合,工具让你的效率爆炸
一句话生成拓扑图!next-ai-draw-io 结合 AI 与 Draw.io,通过自然语言秒出架构图,支持私有部署、免费大模型接口,彻底解放生产力,绘图效率直接爆炸。
772 152