理解 Proxy 和 Object.defineProperty:提升你的 JavaScript 技能(下)

在这里插入图片描述

🤍 前端开发工程师(主业)、技术博主(副业)、已过CET6
🍨 阿珊和她的猫_CSDN个人主页
🕠 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》
🍚 蓝桥云课签约作者、已在蓝桥云课上架的前后端实战课程《Vue.js 和 Egg.js 开发企业级健康管理项目》、《带你从入门到实战全面掌握 uni-app》

文章目录

  • Object.defineProperty 的基本概念和用法
    • 解释 Object.defineProperty 的作用和工作原理
    • 展示如何使用 Object.defineProperty 来定义对象的属性
    • 探讨 Object.defineProperty 的一些常见用例,如数据劫持、属性枚举等
  • 比较 proxy 和 Object.defineProperty
    • 分析两者的相似之处和不同之处
  • 总结
    • 总结 proxy 和 Object.defineProperty 的优势和适用场景

Object.defineProperty 的基本概念和用法

解释 Object.defineProperty 的作用和工作原理

Object.defineProperty() 是 JavaScript 中的一个方法,用于定义或修改对象的属性。
它可以用来设置对象的属性是否可枚举、可读写、可配置以及值的获取和设置函数。

Object.defineProperty() 的语法如下:

Object.defineProperty(obj, prop, descriptor)

其中,obj 是要定义属性的对象,prop 是要定义或修改的属性名,descriptor 是一个对象,用于描述属性的特性。

descriptor 对象的属性如下:

  • configurable:是否可配置,即是否可以通过 delete 操作删除属性,或者通过 Object.defineProperty() 重新定义属性。默认为 false
  • enumerable:是否可枚举,即是否可以通过 for...in 循环遍历属性。默认为 false
  • value:属性的值。如果不指定,则默认为 undefined
  • writable:是否可写,即是否可以通过赋值操作修改属性的值。默认为 false
  • get:获取属性值的函数。当访问该属性时,会调用这个函数返回属性值。
  • set:设置属性值的函数。当通过赋值操作修改属性值时,会调用这个函数。

Object.defineProperty() 的工作原理是:它会在对象的原型链上创建一个新的属性,或者修改已有的属性。如果指定了 value 属性,则会直接设置属性的值;如果指定了 getset 函数,则会将这些函数作为属性的获取和设置方法。

通过使用 Object.defineProperty(),可以实现以下功能:

  • 定义只读属性。
  • 定义不可枚举属性。
  • 定义可配置属性。
  • 定义属性的获取和设置方法。

需要注意的是,Object.defineProperty() 只能定义对象的自有属性,而不能定义继承的属性。如果要定义继承的属性,可以使用 Object.getOwnPropertyDescriptor() 方法获取属性的描述符,然后进行修改。

展示如何使用 Object.defineProperty 来定义对象的属性

Object.defineProperty() 方法可以用来直接在一个对象上定义一个新属性,或者修改一个已存在的属性。下面是一些使用 Object.defineProperty() 的例子:

  1. 定义一个不可枚举、可读写的属性:
const obj = {};// 添加名为 myProperty 的属性,值为 'Hello, World!'
Object.defineProperty(obj, 'myProperty', {value: 'Hello, World!',enumerable: false,writable: true
});console.log(obj.myProperty); 

在这个例子中,Object.defineProperty() 被用来向 obj 对象添加一个名为 myProperty 的新属性。这个属性的值是 ‘Hello, World!’,它是不可枚举的(即,它不会出现在 for...in 循环中),并且是可读写的。

  1. 定义一个只读属性:
const obj = {};// 添加名为 myReadOnlyProperty 的属性,值为 'Read Only'
Object.defineProperty(obj, 'myReadOnlyProperty', {value: 'Read Only',enumerable: true,writable: false
});console.log(obj.myReadOnlyProperty); 
obj.myReadOnlyProperty = '修改失败';
console.log(obj.myReadOnlyProperty); 

在这个例子中,Object.defineProperty() 被用来向 obj 对象添加一个名为 myReadOnlyProperty 的新属性。这个属性的值是 ‘Read Only’,它是可枚举的,但它是只读的,尝试修改它的值会失败。

  1. 定义一个具有 getter 和 setter 的属性:
const obj = {};// 添加名为 myComputedProperty 的属性,值通过 getter 函数计算得出
Object.defineProperty(obj, 'myComputedProperty', {get: function() {return this.name + ' ' + this.age;},enumerable: true
});obj.name = 'John';
obj.age = 30;console.log(obj.myComputedProperty); 

在这个例子中,Object.defineProperty() 被用来向 obj 对象添加一个名为 myComputedProperty 的新属性。这个属性的值是通过一个 getter 函数计算得出的,该函数将对象的 nameage 属性组合在一起。这个属性是可枚举的。

你可以根据需要自定义 Object.defineProperty() 的参数,以满足你对对象属性的特定要求。

探讨 Object.defineProperty 的一些常见用例,如数据劫持、属性枚举等

Object.defineProperty 是 JavaScript 中的一个方法,用于定义或修改对象的属性。它可以用来实现一些常见的用例,如数据劫持、属性枚举等。下面是一些常见的用例:

  1. 数据劫持:通过使用 Object.defineProperty,可以在对象的属性上添加自定义的 getset 方法,从而在获取和设置属性值时进行额外的操作,实现数据劫持。这对于实现观察者模式、响应式编程等非常有用。

    下面是一个简单的数据劫持示例,当修改对象的属性时会触发回调函数:

    // 定义一个对象
    const obj = { data: 10 };// 使用 Object.defineProperty 进行数据劫持
    Object.defineProperty(obj, 'data', {get() {console.log('读取属性值');return this._data;},set(newValue) {console.log('设置属性值');if (newValue > 100) {throw new Error('属性值不能大于 100');}this._data = newValue;}
    });obj.data = 20; 
    console.log(obj.data); 
    
  2. 属性枚举:使用 Object.definePropertyenumerable 属性可以控制属性是否出现在对象的枚举中。默认情况下,对象的属性是可枚举的,但通过将 enumerable 设置为 false,可以使属性不可枚举。

    下面是一个示例,创建一个不可枚举属性:

    const obj = {};// 创建一个不可枚举属性
    Object.defineProperty(obj, 'mySecret', {value: '秘密',enumerable: false
    });// 检查属性是否可枚举
    for (const key in obj) {if (obj.hasOwnProperty(key)) {console.log(key);}
    }console.log(obj.mySecret); 
    
  3. 属性访问控制:除了 getset 方法外,Object.defineProperty 还可以使用 configurable 属性来控制属性的可配置性。将 configurable 设置为 false 可以防止通过 delete 操作删除属性,或者通过 Object.defineProperty 重新定义属性。

    下面是一个示例,创建一个不可配置的属性:

    const obj = {};// 创建一个不可配置属性
    Object.defineProperty(obj, 'myProtected', {value: '受保护的',configurable: false
    });try {// 尝试删除属性delete obj.myProtected;console.log('属性已删除');
    } catch (error) {console.log('删除属性失败:', error.message);
    }try {// 尝试重新定义属性Object.defineProperty(obj, 'myProtected', {value: '新值'});console.log('属性已重新定义');
    } catch (error) {console.log('重新定义属性失败:', error.message);
    }console.log(obj.myProtected); 
    

这些是 Object.defineProperty 的一些常见用例。它提供了一种灵活的方式来操作对象的属性,实现一些高级的功能,如数据劫持、属性枚举控制和属性访问控制。

比较 proxy 和 Object.defineProperty

分析两者的相似之处和不同之处

Proxy和Object.defineProperty都是JavaScript中用于拦截对象操作的机制,它们有相似之处,也有不同之处,下面对它们进行比较和分析。

相似之处:

  1. 都可以对对象进行拦截
  2. 都可以拦截属性的读、写和删除操作
  3. 都可以在拦截操作时执行自定义逻辑
  4. 都可以阻止默认操作或修改默认操作的结果。
  5. 都可以监听对象属性变化,以及拦截原型属性访问。
  6. 都可以控制对象的可枚举性、可配置性和可写性。

在这里插入图片描述

不同之处:

  1. Object.defineProperty只能拦截单个属性,而Proxy可以拦截整个对象。
  2. Proxy对拦截的操作范围更广,可以拦截数组操作和函数调用,而Object.defineProperty只能对属性访问进行拦截
  3. Proxy可以动态添加拦截器,而Object.defineProperty直接对属性进行设置
  4. Proxy在处理某些情况下的性能可能会比Object.defineProperty差,特别是当拦截大量属性时。
  5. Object.defineProperty是ES5中引入的,而Proxy是ES6中引入的。

在实际使用中,Object.defineProperty通常用于具体属性的拦截和操作,例如属性的监听,数据劫持等;而Proxy则更适用于对整个对象的拦截和处理,例如实现面向切面编程、缓存、转发等高级功能。但需要注意的是,Proxy虽然功能十分强大,但是并不是所有浏览器都支持,因此在开发时需要根据使用场景和项目需求做出合适的选择,避免因为技术兼容性问题导致项目无法实现。

总结

总结 proxy 和 Object.defineProperty 的优势和适用场景

以下是使用表格总结Proxy和Object.defineProperty的优势和适用场景的示例:

功能ProxyObject.defineProperty
直接拦截对对象的操作
支持拦截的操作范围广泛
可以拦截的操作包括读取、写入、删除等
可以自定义拦截操作的逻辑
可以拦截数组的操作
支持动态添加和删除拦截器
支持对整个对象进行拦截
可以修改拦截的操作结果
替代Object.defineProperty的用法部分替代
对象引用改变是否需要重新设置拦截器需要重新设置不需要重新设置

适用场景:

  • Proxy适用于需要对整个对象进行拦截且拦截的操作范围广泛的场景。它允许自定义拦截逻辑,可以用来实现数据绑定、验证、日志记录等功能。
  • Object.defineProperty适用于只需要对对象的某个属性进行拦截的场景。它可以用来实现对属性的读、写、删除操作进行定制,例如监听属性变化、冻结对象等。

请注意,Proxy是ES6引入的新特性,对于一些需要兼容老版本浏览器的项目可能不适合使用。而Object.defineProperty(作为ES5的特性)在老版本浏览器中也可能存在一些限制。因此,在选择使用哪种拦截机制时,你应该根据具体的项目需求和兼容性考虑做出决策。

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

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

相关文章

IDEA利用插件完成properties与yml的互相转换(mac与wins通用)

步骤一、插件安装 点击屏幕左上方的IDEA,然后点击Preferences(相当于wins里的settings) 进入后点击Plugins,在插件商城中搜索并安装 Convert YAML and Properties File 这个插件 二、使用 右键选择你需要转换的配置文件,选择Convert YAML …

HarmonyOS给应用添加视频播放功能

Video组件的使用 概述 在手机、平板或是智慧屏这些终端设备上,媒体功能可以算作是我们最常用的场景之一。无论是实现音频的播放、录制、采集,还是视频的播放、切换、循环,亦或是相机的预览、拍照等功能,媒体组件都是必不可少的。…

JAVA高级(后端需深入移步)

单元测试:使用Junit单元测试框架 使用Junit单元测试: 通过左侧的对❌来进行提示 Junit框架的常见注解: 反射(用于框架,也是最重要):展示框架的成员信息 由于是用于对象,即使在获取…

CDN是如何实现网络加速的?

最近很多网站用户都在谈论香港服务器的同时,也在讨论CDN加速服务的相关话题,谈论的目的就是想通过相关的设置和服 务来有一个好的上网体验,同时减少同行DDOS的攻击。 那么,对网站进行CDN加速服务到底能够让网络实现哪些功能呢&am…

低代码开发:数字化“装配线”的崛起

一、什么是低代码 首先,我们不妨从低代码开发的定义和起源说起。低代码开发(Low Code Development),指通过可视化的图形界面来进行应用程序开发和部署的方式,大大降低了开发门槛,用户不需要掌握传统文本编程语言就可以进行开发。 低代码开发起源于20世纪90年代,经历了3个发展…

力扣300. 最长递增子序列

动态规划 思路: 假设 dp[i] 为前 i 个元素构成的最长递增子序列的个数,包含 nums[i];则 dp[i] 构成序列上一个元素 nums[j] 构成最长递增子序列 dp[j],则 dp[i] dp[j] 1;如果动态取 j ∈ [0, i - 1],则选…

Github入门

简介 github是一个基于git的代码仓库,可以通过git来上传和下载代码。国内类似的有gitee。 开源项目一般会申明开源协议。我们可以基于可商用的代码开发我们自己的项目,以期进行快速开发。 一般情况下gitee上的项目基本都够我们使用了。 git基础 Git…

二叉树(接口函数的实现)

今天继续来分享的是二叉树,我们废话不多说,直接来看下面的几个接口函数,然后我们把他们实现,我们就掌握二叉树的二分之一(今天粉丝破千了,属实有点高兴了)。 typedef char BTDataType;typedef s…

OSG加载地形

这是网上下的一个代码; 先看一下代码; KeyboardHandler.h; #ifndef KEYBOARD_HANDLER_H #define KEYBOARD_HANDLER_H #include <iostream> #include <osgGA/GUIEventHandler>class keyboardEventHandler : public osgGA::GUIEventHandler { public:typedef v…

HTML插入视频和音频(详解)

&#x1f4cd;文章目录&#x1f4cd; &#x1f9c0;一&#xff0c;简介&#x1f9c0;二&#xff0c;视频(video)&#x1f367;1&#xff0c;普通的视频插入&#x1f367;2&#xff0c;在html5中嵌入视频网站视频 &#x1f9c0;三&#xff0c;音频(audio) &#x1f9c0;一&#…

linux中的od命令与hexdump命令

初步解读两个命令 在Linux中&#xff0c;"od"和"hexdump"命令都用于以十六进制和其他格式显示文件的内容。它们提供了对文件进行二进制查看和分析的功能。以下是它们的简要说明&#xff1a; od命令&#xff1a; “od”&#xff08;octal dump&#xff09;…

德语 Alt 代码表

德语的 Alt 代码表&#xff0c;请参考下图。 输入方法就是按住 Alt 键不松开&#xff0c;然后在小键盘上输入字符&#xff0c;松开 Alt 键&#xff0c;计算机就能输出上面的字符了。 德语 Alt 代码表 - 系统容器 - iSharkFly德语的 Alt 代码表&#xff0c;请参考下图。 输入方…

Spring Security(一)架构概览

一、 Spring Security 架构概览 1. Spring Security 简介 在Java企业级开发中&#xff0c;安全管理方面的框架非常少&#xff0c;一般来说&#xff0c;主要是三种方案&#xff1a; ShiroSpring Security开发者自己实现 Spring Security基于Spring框架&#xff0c;提供了一套…

Java常用注解

文章目录 第一章、Java注解与元数据1.1&#xff09;元数据与注解概念介绍1.2&#xff09;Java注解的作用和使用1.3&#xff09;注解的分类 第二章、Mybatis框架常用注解2.1&#xff09;Mybatis注解概览2.2&#xff09;常用注解MapperScanMapperSelectInsertUpdateDeleteParam结…

学习openAI 短长期AGI计划、使命、宪章、开创性研究、产品、工作待遇等

网站的设计&#xff1a;简洁而现代 主页 使命&#xff1a;Creating safe AGI that benefits all of humanity. &#xff08;比人类更聪明的人工智能系统&#xff09;&#xff08;自己实现或帮别人实现都认为是达成使命&#xff09;&#xff08;造福全人类&#xff1a;最大限…

springboot笔记

1、springboot中的缓存标签Cacheable使用场景 Cacheable 只是为了让你省略掉是使用集合来保存缓存数据的代码&#xff0c;给你的业务代码横切入缓存的逻辑.然后使用缓存的场景就是读多写少的场景&#xff0c;读操作特别频繁的话&#xff0c;还是把热点数据缓存起来&#xff0c…

36V/48V转12V 10A直流降压DC-DC芯片-AH1007

AH1007是一款36V/48V转12V 10A直流降压&#xff08;DC-DC&#xff09;芯片&#xff0c;它是一种高性能的降压变换器&#xff0c;常用于工业、汽车和电子设备等领域。 AH1007采用了先进的PWM调制技术和开关电源控制算法&#xff0c;能够高效地将输入电压从36V/48V降低到12V&…

MATLAB实现图像变换和滤波

MATLAB实现图像变换和滤波方法对具有不同特征的灰度图像进行处理 图像变换方法包括&#xff1a;DFT及IDFT&#xff0c;DCT及IDCT 图像滤波方法包括低通滤波和高通滤波 图像变换 DFT/IDFT 图像一般是二维的&#xff0c;根据二维离散傅里叶变换公式DFT&#xff0c;可以将图片…

如何使用Docker将.Net6项目部署到Linux服务器(一)

目录 配置服务器环境 配置yum 配置docker 安装.NetCore SDK6.0 发布Net6 添加Dockerfile。 发布文件。 编辑DockerFile文件 ​编辑 上传文件 安装MySql 配置服务器环境 配置yum 在配置yum之前&#xff0c;我们需要先了解yum是什么&#xff0c;yum&#xff0c;是Yellow…

力扣日记12.13-【二叉树篇】从中序与后序遍历序列构造二叉树

力扣日记&#xff1a;【二叉树篇】从中序与后序遍历序列构造二叉树 日期&#xff1a;2023.12.13 参考&#xff1a;代码随想录、力扣 106. 从中序与后序遍历序列构造二叉树 题目描述 难度&#xff1a;中等 给定两个整数数组 inorder 和 postorder &#xff0c;其中 inorder 是二…