JavaScript拦截工具Proxy

在 JavaScript 中,Proxy 是一种强大的工具,它允许你通过拦截对象的基本操作(如属性访问、赋值、函数调用等)来定义自定义行为。Proxy 是在 ECMAScript 6(ES6)中引入的,主要用于增强对象的功能和行为。


基本语法

const proxy = new Proxy(target, handler);
  • target:需要代理的目标对象(可以是普通对象、数组或另一个 Proxy)。
  • handler:一个对象,包含特定操作的拦截方法(称为“trap”)。

常用 handler 的拦截方法(Trap)

以下是 Proxy 支持的常用 trap 方法,以及它们的用途:

Trap用途
get拦截属性读取,例如 proxy.prop
set拦截属性赋值,例如 proxy.prop = value
has拦截属性检查,例如 'prop' in proxy
deleteProperty拦截属性删除,例如 delete proxy.prop
ownKeys拦截对象自身属性的枚举,例如 Object.keys(proxy)for...in 循环。
apply拦截函数调用,例如 proxy()
construct拦截构造函数调用,例如 new proxy()

常见用法示例

1. 属性读取拦截
const person = {name: "Alice",age: 25
};const proxy = new Proxy(person, {get(target, prop) {if (prop in target) {return target[prop];} else {return `Property "${prop}" does not exist.`;}}
});console.log(proxy.name); // 输出: Alice
console.log(proxy.gender); // 输出: Property "gender" does not exist.

2. 属性赋值拦截(数据验证)
const person = {name: "Alice",age: 25
};const proxy = new Proxy(person, {set(target, prop, value) {if (prop === "age" && typeof value !== "number") {throw new TypeError("Age must be a number.");}target[prop] = value;return true; // 必须返回 true,表示成功}
});proxy.age = 30; // 正常赋值
console.log(proxy.age); // 输出: 30proxy.age = "thirty"; // 抛出错误: Age must be a number

3. 日志记录
const person = {name: "Alice",age: 25
};const proxy = new Proxy(person, {get(target, prop) {console.log(`Getting property "${prop}"`);return target[prop];},set(target, prop, value) {console.log(`Setting property "${prop}" to "${value}"`);target[prop] = value;return true;}
});console.log(proxy.name); // 输出日志并获取值
proxy.age = 26; // 输出日志并更新值

4. 禁止删除属性
const person = {name: "Alice",age: 25
};const proxy = new Proxy(person, {deleteProperty(target, prop) {if (prop === "name") {throw new Error("Cannot delete 'name'");}delete target[prop];return true;}
});delete proxy.age; // 成功
console.log(proxy); // 输出: { name: 'Alice' }delete proxy.name; // 抛出错误: Cannot delete 'name'

5. 拦截数组操作
const numbers = [1, 2, 3];const proxy = new Proxy(numbers, {get(target, prop) {if (prop === "last") {return target[target.length - 1];}return target[prop];},set(target, prop, value) {if (prop === "length" && value < target.length) {throw new Error("Cannot shrink the array.");}target[prop] = value;return true;}
});console.log(proxy.last); // 输出: 3
proxy.push(4); // 正常操作
console.log(proxy); // 输出: [1, 2, 3, 4]proxy.length = 2; // 抛出错误: Cannot shrink the array.

6. 动态计算属性
const calculator = {factor: 2
};const proxy = new Proxy(calculator, {get(target, prop) {if (prop === "double") {return target.factor * 2;}return target[prop];}
});console.log(proxy.factor); // 输出: 2
console.log(proxy.double); // 输出: 4

总结

Proxy 提供了一种灵活的方式来拦截和自定义对象的行为,适用于以下场景:

  • 数据验证与保护
  • 动态属性计算
  • 日志记录或调试
  • 实现代理模式或权限控制

通过 Proxy,可以让代码更具可读性和功能性,同时也需要注意避免过度复杂的拦截逻辑。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/887172.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

HAProxy面试题及参考答案(精选80道面试题)

目录 什么是 HAProxy? HAProxy 主要有哪些功能? HAProxy 的关键特性有哪些? HAProxy 的主要功能是什么? HAProxy 的作用是什么? 解释 HAProxy 在网络架构中的作用。 HAProxy 与负载均衡器之间的关系是什么? HAProxy 是如何实现负载均衡的? 阐述 HAProxy 的四层…

使用flink编写WordCount

1. env-准备环境 2. source-加载数据 3. transformation-数据处理转换 4. sink-数据输出 5. execute-执行 流程图&#xff1a; DataStream API开发 //nightlies.apache.org/flink/flink-docs-release-1.13/docs/dev/datastream/overview/ 添加依赖 <properties>&l…

# issue 4 进程控制函数

目录 一、进程控制函数一 二、进程控制函数二 启动进程&#xff1a;&#xff08;exec系列&#xff09; 创建新进程&#xff1a; 测试代码&#xff1a; 测试结果&#xff1a; 三、进程控制函数三 结束进程&#xff1a; 测试代码&#xff1a; 测试结果&#xff1a; 四、…

信息系统项目管理师——第7章 项目立项管理 笔记

7项目立项管理 投资前时期的四个阶段&#xff1a;立项申请、初步可行性研究、详细可行性研究、评估与决策 详细可行性研究是不可缺少的。 项目建议书是项目发展周期的初始阶段&#xff0c;是国家或上级主管部门选择项目的依据&#xff0c;也是可行性研究的依据。 项目建议书核心…

Java项目实战II基于SpringBoot的共享单车管理系统开发文档+数据库+源码)

目录 一、前言 二、技术介绍 三、系统实现 四、核心代码 五、源码获取 全栈码农以及毕业设计实战开发&#xff0c;CSDN平台Java领域新星创作者&#xff0c;专注于大学生项目实战开发、讲解和毕业答疑辅导。获取源码联系方式请查看文末 一、前言 在共享经济蓬勃发展的今天…

Linux 网络编程之UDP套接字

前言 前面我们对网络的发展&#xff0c;网络的协议、网路传输的流程做了介绍&#xff0c;最后&#xff0c;我们还介绍了 IP 和 端口号&#xff0c;ip port 叫做 套接字 socket&#xff0c; 本期我们就来介绍UDP套接字编程&#xff01; 目录 1、预备知识 1.1 传输层协议: T…

---Arrays类

一 java 1.Arrays类 1.1 toString&#xff08;&#xff09; 1.2 arrays.sort( )-----sort排序 1&#xff09;直接调用sort&#xff08;&#xff09; Arrays.sort() 方法的默认排序顺序是 从小到大&#xff08;升序&#xff09;。 2&#xff09;定制排序【具体使用时 调整正负…

Java 对象头、Mark Word、monitor与synchronized关联关系以及synchronized锁优化

1. 对象在内存中的布局分为三块区域&#xff1a; &#xff08;1&#xff09;对象头&#xff08;Mark Word、元数据指针和数组长度&#xff09; 对象头&#xff1a;在32位虚拟机中&#xff0c;1个机器码等于4字节&#xff0c;也就是32bit&#xff0c;在64位虚拟机中&#xff0…

6.7机器学习期末复习题

空间 样本空间 就是属性的所有可能情况&#xff0c;包括了一切可能出现或不可能出现的所有样本情况 版本空间&假设空间 假设空间就是在样本空间的基础上&#xff0c;给所有属性都加了一个通配符&#xff0c;表示任意即可&#xff1b;以及加上了一个空集&#xff0c;表示…

数据结构与算法——1122—复杂度总结检测相同元素

1、复杂度总结 1、时间复杂度计算遵循的原则 1、复杂度与其具体的常系数无关&#xff08;即&#xff1a;常数项的系数不要&#xff09; 2、多项式级复杂度相加的时候&#xff0c;把其高项作为结果&#xff08;即&#xff1a;多项式只保留最大项&#xff09; 3、O(1)含义为&…

Qt界面设计时使各控件依据窗口缩放进行栅格布局的方法

图1 最终效果 想要达成上述图片的布局效果&#xff0c;具体操作如下&#xff1a; 新建一窗体&#xff1a; 所需控件如下&#xff1a; Table View控件一个&#xff1b; Group Box控件一个&#xff1b; Push Button控件2个&#xff1b; Horiziontal Spacer控件2个&#xf…

mac安装Pytest、Allure、brew

安装环境 安装pytest 命令 pip3 install pytest 安装allure 命令&#xff1a;brew install allure 好吧 那我们在安装allure之前 我们先安装brew 安装brew 去了官网复制了命令 还是无法下载 如果你们也和我一样可以用这个方法哦 使用国内的代码仓库来执行brew的安装脚本…

数据结构C语言描述5(图文结合)--队列,数组、链式、优先队列的实现

前言 这个专栏将会用纯C实现常用的数据结构和简单的算法&#xff1b;有C基础即可跟着学习&#xff0c;代码均可运行&#xff1b;准备考研的也可跟着写&#xff0c;个人感觉&#xff0c;如果时间充裕&#xff0c;手写一遍比看书、刷题管用很多&#xff0c;这也是本人采用纯C语言…

一篇文章了解机器学习

一篇文章了解机器学习&#xff08;上&#xff09; 一、软件版本安装二、数据集的加载三、数据集的切分四、数据特征提取及标准化1、字典数据的特征提取2、文本特征向量的提取3、数据标准化处理 四、特征降维注&#xff1a;训练器的区别&#xff1a;&#xff1a;五、模型的训练与…

day03(单片机高级)RTOS

目录 RTOS(实时操作系统) 裸机开发模式 轮询方式 前后台&#xff08;中断方式&#xff09; 改进&#xff08;前后台&#xff08;中断&#xff09;&#xff09;定时器 裸机进一步优化 裸机的其他问题 RTOS的概念 什么是RTOS 为什么要使用 RTOS RTOS的应用场景 RTOS的…

Hello-Go

Hello-Go 环境变量 GOPATH 和 GOROOT &#xff1a;不同于其他语言&#xff0c;go中没有项目的说法&#xff0c;只有包&#xff0c;其中有两个重要的路径&#xff0c;GOROOT 和 GOPATH Go开发相关的环境变量如下&#xff1a; GOROOT&#xff1a;GOROOT就是Go的安装目录&…

pytorch官方FasterRCNN代码详解

本博文转自捋一捋pytorch官方FasterRCNN代码 - 知乎 (zhihu.com)&#xff0c;增加了其中代码的更详细的解读&#xff0c;以帮助自己理解该代码。 代码理解的参考Faster-RCNN全面解读(手把手带你分析代码实现)---前向传播部分_手把手faster rcnn-CSDN博客 1. 代码结构 作为 to…

SpringBootTest启动时出现循环依赖问题

在公司项目开发中由于SpringBoot启动类配置了setAllowCircularReferences为true在特定的业务逻辑下需要该配置&#xff0c;但我们需要使用SpringBootTest单元测试的时候引入我们开发的配置文件发现不生效&#xff0c; 解决方法&#xff1a; SpringBootTest(properties "…

力扣11.22

44. 通配符匹配 给你一个输入字符串 (s) 和一个字符模式 &#xff0c;请你实现一个支持 ‘?’ 和 ‘*’ 匹配规则的通配符匹配&#xff1a; ‘?’ 可以匹配任何单个字符。‘*’ 可以匹配任意字符序列&#xff08;包括空字符序列&#xff09;。 判定匹配成功的充要条件是&a…

@RequestBody和前端的关系以及,如何在前后端之间传递数据?

RequestBody 注解在 Spring MVC 中用于将 HTTP 请求体中的数据绑定到控制器方法的参数上。为了更好地理解 RequestBody 和前端之间的关系&#xff0c;我们可以从以下几个方面进行探讨&#xff1a; 1. 请求体的格式 前端发送的请求体通常是一个 JSON 字符串&#xff0c;也可以…