Object.defineProperty 和 Proxy 使用例子

Object.defineProperty 是 JavaScript 中定义或修改对象属性的工具。它允许开发者更精确地控制属性的行为,如是否可枚举、是否可写、是否可配置等。

核心特性描述符
  • value: 属性的值,默认值为 undefined
  • writable: 布尔值,表示属性值是否可修改,默认值为 false
  • enumerable: 布尔值,表示属性是否可在对象的枚举属性中被枚举,默认值为 false
  • configurable: 布尔值,表示属性描述符是否可被修改,属性是否可被删除,默认值为 false
  • get: 返回属性值的函数,无默认值。取值器函数是用于访问属性的逻辑,可以定义自定义的读取逻辑。
  • set: 设置属性值的函数,无默认值。设置器函数是用于写入属性的逻辑,可以定义自定义的写入逻辑。
典型应用
  • 控制属性的可写性: 确保某些属性不可被意外修改。
  • 控制属性的可枚举性: 隐藏属性,使其在 for...in 循环和 Object.keys 中不可见。
  • 控制属性的可配置性: 锁定属性,防止其被删除或重新定义。
  • 创建访问器属性: 使用 getset 函数定义属性的访问和设置逻辑。
使用方法
Object.defineProperty(obj, prop, descriptor);
  • obj: 目标对象。
  • prop: 要定义或修改的属性的名称。
  • descriptor: 包含属性描述符的对象。

Proxy 

Proxy 是 ECMAScript 6 (ES6) 引入的一种用于定义自定义行为的对象。它允许开发者拦截和自定义基本操作,例如属性访问、赋值、枚举、函数调用等。

核心概念
  • target: 被代理的目标对象,可以是任何类型的对象(对象、数组、函数等)。
  • handler: 包含捕捉器(拦截方法)的对象。这些捕捉器可以拦截和自定义对 target 的各种操作。
常用捕捉器
  • get: 拦截属性读取操作,允许开发者定义读取属性时的行为。
  • set: 拦截属性设置操作,允许开发者定义写入属性时的行为。
  • has: 拦截 in 操作符,允许开发者定义检查属性是否存在时的行为。
  • deleteProperty: 拦截 delete 操作符,允许开发者定义删除属性时的行为。
  • apply: 拦截函数调用操作,允许开发者定义调用函数时的行为。
  • construct: 拦截 new 操作符,允许开发者定义实例化对象时的行为。
典型应用
  • 日志和调试: 在访问或修改对象属性时记录日志。
  • 数据验证: 在设置属性值时进行验证和检查。
  • 虚拟化和延迟计算: 按需计算属性值或延迟执行某些操作。
  • 安全和访问控制: 限制或控制对对象属性的访问。
使用方法
new Proxy(target, handler);
  • target: 要包装的目标对象。
  • handler: 包含捕捉器的对象,用于定义拦截和自定义操作的逻辑。

比较与选择

  • Object.defineProperty:

    • 提供精确控制对象属性的特性。
    • 适合需要严格控制单个对象属性行为的场景。
  • Proxy:

    • 提供全面的拦截和自定义功能。
    • 适合需要对整个对象的所有操作进行控制和自定义的场景。

在实际应用中,Proxy 提供了更强大的功能和更高的灵活性,但 Object.defineProperty 仍然是一个非常有用的工具,尤其是在需要简单地控制某些属性行为时。

使用 Object.defineProperty 的例子

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Object.defineProperty Example</title>
</head>
<body>
    <div id="nameDisplay">Name: </div>
    <div id="ageDisplay">Age: </div>

    <script>
       let person = {
    _name: 'Alice',
    _age: 25
};

Object.defineProperty(person, 'name', {
    get() {
        return this._name;
    },
    set(newValue) {
        this._name = newValue;
        document.getElementById('nameDisplay').innerText = `Name: ${newValue}`;
    },
    enumerable: true,
    configurable: true
});

Object.defineProperty(person, 'age', {
    get() {
        return this._age;
    },
    set(newValue) {
        this._age = newValue;
        document.getElementById('ageDisplay').innerText = `Age: ${newValue}`;
    },
    enumerable: true,
    configurable: true
});

// 初始化 DOM 内容
document.getElementById('nameDisplay').innerText = `Name: ${person.name}`;
document.getElementById('ageDisplay').innerText = `Age: ${person.age}`;

// 测试更改属性
setTimeout(() => {
    person.name = 'Bob';
    person.age = 30;
}, 2000);

    </script>
</body>
</html>

使用 Proxy 的例子

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Proxy Example</title>
</head>
<body>
    <div id="nameDisplay">Name: </div>
    <div id="ageDisplay">Age: </div>

    <script>
        // 定义 person 对象
        let person = {
            name: 'Alice',
            age: 25
        };

        // 创建一个 Proxy 以监听属性变化
        const handler = {
            get(target, property) {
                // 返回属性值
                return target[property];
            },
            set(target, property, value) {
                // 更新属性值
                target[property] = value;

                // 更新 DOM
                if (property === 'name') {
                    document.getElementById('nameDisplay').innerText = `Name: ${value}`;
                } else if (property === 'age') {
                    document.getElementById('ageDisplay').innerText = `Age: ${value}`;
                }

                // 返回 true 表示设置成功
                return true;
            }
        };

        // 创建代理对象
        const proxyPerson = new Proxy(person, handler);

        // 初始设置 DOM 内容
        document.getElementById('nameDisplay').innerText = `Name: ${proxyPerson.name}`;
        document.getElementById('ageDisplay').innerText = `Age: ${proxyPerson.age}`;

        // 测试读取和更改属性
        setTimeout(() => {
            console.log(proxyPerson.name); // 读取属性值
            console.log(proxyPerson.age);

            proxyPerson.name = 'Bob'; // 设置属性值
            proxyPerson.age = 30;
        }, 2000);
    </script>
</body>
</html>

解释

  1. Object.defineProperty 示例:

    • 通过 Object.defineProperty 定义 person 对象的 nameage 属性。
    • 使用 getset 方法分别定义属性的读取和设置行为。
    • 属性值变化时,更新对应的 DOM 元素。
  2. Proxy 示例:

    • 通过 Proxy 创建 person 对象的代理 proxyPerson
    • 使用 get 方法拦截属性读取操作,返回属性值。
    • 使用 set 方法拦截属性设置操作,更新属性值并更新 DOM 元素。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【最大公约数 唯一分解定理 调和级数】2862. 完全子集的最大元素和

本文涉及知识点 质数、最大公约数、菲蜀定理 组合数学汇总 唯一分解定理 调和级数 LeetCode2862. 完全子集的最大元素和 给你一个下标从 1 开始、由 n 个整数组成的数组。你需要从 nums 选择一个 完全集&#xff0c;其中每对元素下标的乘积都是一个 完全平方数&#xff0c;例…

程序员学CFA——数量分析方法(六)

数量分析方法&#xff08;六&#xff09; 假设检验假设检验的步骤假设检验的基本思想与步骤估计与假设检验的区别假设检验的基本思想假设检验的步骤 假设检验的相关概念原假设与备择假设检验统计量及其分布显著性水平双尾检验与单尾检验p值第一类错误与第二类错误统计显著与经济…

力扣HOT100 - 155. 最小栈

解题思路&#xff1a; 辅助栈 class MinStack {private Stack<Integer> stack;private Stack<Integer> min_stack;public MinStack() {stack new Stack<>();min_stack new Stack<>();}public void push(int val) {stack.push(val);if (min_stack.i…

SpringBoot集成jxls2实现复杂(多表格)excel导出

核心依赖 需求 导出多个表格&#xff0c;包含图片&#xff0c;类似商品标签 1.配置模板 创建一个xlsx的模板文件&#xff0c;配置如下 该模板进行遍历了两次&#xff0c;因为我想要导出的数据分为两列展示&#xff0c;左右布局&#xff0c;一个循环实现不了&#xff0c;所以采…

计算机系列之面向对象、设计模式

24、面向对象技术&#xff08;重要&#xff0c;10分左右&#xff09; 1、面向对象开发 (1)对象:由数据及其操作所构成的封装体&#xff0c;是系统中用来描述客观事务的个实体&#xff0c;是构成系统的一个基本单位。一个对象通常可以由对象名、属性和方法3个部分组成。 (2)类…

YOLOV5更换转置卷积,助力涨点!

由于转置卷积是nn库自带的,所以我们直接找到models文件夹中的yolo.py文件中的 parse_model函数,再在如下图的地方添加转置卷积模块 # YOLOv5 🚀 by Ultralytics, AGPL-3.0 license """ YOLO-specific modules.Usage:$ python models/yolo.py --cfg yolov5s.…

ARM 交叉编译搭建SSH

一、源码下载 zlib&#xff1a;zlib-1.3.1.tar.xz openssl&#xff1a;openssl-0.9.8d.tar.gz openssh&#xff1a;openssh-4.6p1.tar.gz 二、交叉编译 1、zlib 编译参考这里 2、openssl tar -xf openssl-0.9.8d.tar.gz ./Configure --prefix/opt/ssh/openssl os/compile…

一对一WebRTC视频通话系列(五)——综合调试和功能完善

本系列博客主要记录一对一WebRTC视频通话实现过程中的一些重点&#xff0c;代码全部进行了注释&#xff0c;便于理解WebRTC整体实现。 本专栏知识点是通过<零声教育>的音视频流媒体高级开发课程进行系统学习&#xff0c;梳理总结后写下文章&#xff0c;对音视频相关内容感…

猿匹配,一款使用环信实现的一个开源聊天应用含服务器

前言 之前写了一篇Android开发集成聊天环信SDK3.x简单开始&#xff0c;然后最近得空开发了一款使用环信实现的实时聊天应用&#xff0c;包含简单的服务器端&#xff0c;并开源给大家&#xff0c;有兴趣的同学可以一起搞一下&#xff0c;详细介绍看下边吧 上代码 服务器&#…

文章解读与仿真程序复现思路——电力自动化设备EI\CSCD\北大核心《计及全生命周期成本的公交光伏充电站储能优化配置方法》

本专栏栏目提供文章与程序复现思路&#xff0c;具体已有的论文与论文源程序可翻阅本博主免费的专栏栏目《论文与完整程序》 论文与完整源程序_电网论文源程序的博客-CSDN博客https://blog.csdn.net/liang674027206/category_12531414.html 电网论文源程序-CSDN博客电网论文源…

清华团队开发首个AI医院小镇模拟系统;阿里云发布通义千问 2.5:超越GPT-4能力;Mistral AI估值飙升至60亿美元

&#x1f989; AI新闻 &#x1f680; 清华团队开发首个AI医院小镇模拟系统 摘要&#xff1a;来自清华的研究团队最近开发出了一种创新的模拟系统&#xff0c;名为"Agent Hospital"&#xff0c;该系统能够完全模拟医患看病的全流程&#xff0c;其中包括分诊、挂号、…

【八十五】【算法分析与设计】单调栈的全新版本,两个循环维护左小于和右小于信息,84. 柱状图中最大的矩形,85. 最大矩形

84. 柱状图中最大的矩形 给定 n 个非负整数&#xff0c;用来表示柱状图中各个柱子的高度。每个柱子彼此相邻&#xff0c;且宽度为 1 。 求在该柱状图中&#xff0c;能够勾勒出来的矩形的最大面积。 示例 1: 输入&#xff1a;heights [2,1,5,6,2,3] 输出&#xff1a;10 解释&am…

Go的安装与配置

安装 https://go.dev/dl/ 以Windows上安装为例&#xff1a; 下一步下一步&#xff0c;记住安装位置 安装完成后 win rcmd 键入go version检查是否安装成功 配置 Go 工作区 Go 在组织项目文件方面与其他编程语言不同。 Go 是在工作区的概念下工作的。但是从版本 1.11 开始&…

docker-compose部署java项目

docker-compose是定义和运行多容器的工具。换句话说就是通过配置yml文件来运行容器&#xff0c;简化了每次输入docker run等命令&#xff0c;把这些命令配置在yml文件统一管理&#xff0c;而且可以用一个yml文件一次启动多个容器&#xff0c;启动时还可以设置各个容器的依赖关系…

远程开机与远程唤醒BIOS设置

远程开机与远程唤醒BIOS设置 在现代计算机应用中&#xff0c;远程管理和控制已成为许多企业和个人的基本需求。其中&#xff0c;远程开机和远程唤醒是两项非常实用的功能。要实现这些功能&#xff0c;通常需要在计算机的BIOS中进行一些特定的设置。以下是对远程开机和远程唤醒…

如何判断nat网络?如何内网穿透

大家都清楚&#xff0c;如果你想开车&#xff0c;就必须要给车上一个牌照&#xff0c;随着车辆越来越多&#xff0c;为了缓解拥堵&#xff0c;就需要摇号&#xff0c;随着摇号的人数越来越多&#xff0c;车牌对于想开车的人来说已经成为奢望。在如今的IPv4时代&#xff0c;我们…

全自动减压器法二氧化碳气容量测试仪:饮料行业的革新与未来

全自动减压器法二氧化碳气容量测试仪&#xff1a;饮料行业的革新与未来 一、引言 在追求品质与效率的现代饮料生产领域&#xff0c;全自动减压器法二氧化碳气容量测试仪凭借其高精度、高效率及无人工干预的显著优势&#xff0c;正逐渐成为行业的标杆。特别是在碳酸饮料的生产中…

USB系列五:USB设备配置(重要)

在USB总线接口协议中&#xff0c;对于USB外部设备功能特征是通过端点&#xff08;Endpoint&#xff09;、配置&#xff08;Configuration&#xff09;和接口&#xff08;Interface&#xff09;来描述的&#xff0c;这些就是典型的USB描述符。 USB主机通过设备请求来读取外部设…

并行执行线程资源管理方式——《OceanBase 并行执行》系列 3

在某些特定场景下&#xff0c;由于需要等待线程资源&#xff0c;并行查询会遇到排队等待的情况。本篇博客将介绍如何管理并行执行线程资源&#xff0c;以解决这种问题。 《OceanBase并行执行》系列的内容分为七篇博客&#xff0c;本篇是其中的第三篇。前2篇如下&#xff1a; 一…

分布式与一致性协议之Quorum NWR算法

Quorum NWR算法 概述 不知道你在工作中有没有遇到过这样的事情:你开发实现了一套AP型分布式系统&#xff0c;实现了最终一致性&#xff0c;且业务接入后运行正常&#xff0c;一切看起来都那么美好。 可是突然有同事说&#xff0c;我们要拉这几个业务的数据做实时分析&#xf…
最新文章