Object.defineProperty() 详解

news2024/11/18 13:32:10

一、对象的定义与赋值

我们经常使用的定义与赋值方法 obj.xxx = value 或 obj['xxx'] = value,并且可以定义任意类型的值,如下所示:

let obj = {};
obj.name = 'bjl';
obj['age'] = 18;
obj.sayHi = function() {console.log('Hi')};
console.log(obj)  // {name: 'bjl', age: 18, sayHi: fn}

二、Object.defineProperty() 语法

Object.defineProperty() 的作用就是直接在一个对象上定义新属性,或者修改一个已经存在的属性。语法格式为:

Object.defineProperty(object, propName, descriptor);
  1. object:需要定义属性的当前对象
  2. propName:当前需要定义的属性名
  3. descriptor:属性描述符

三、属性描述符

都有哪些属性描述符呢?如下:

  • value:设置属性的值
  • writable:值是否可以重写
  • set:目标属性设置值的方法
  • get:目标属性获取值的方法
  • enumerable:目标属性是否可以被枚举(是否可以遍历)
  • configurable:目标属性是否可以被删除或是否可以再次修改特性

通过 Object.defineProperty() 为对象定义属性有两种形式,但不能混合使用,分别是数据描述符、存取描述符。

1. 数据描述符(value、writable)

当使用了 value 和 writable 属性,不允许使用 getter 或 setter 这两个方法。

  • value:设置属性值
let obj = {};
Object.defineProperty(obj, 'name', {
    value: 'bjl'
})
console.log(obj.name)  // bjl
  • writable:描述对象是否可写,默认值为false,表示只读属性
let obj = {};
Object.defineProperty(obj, 'name', {
    value: 'bjl'
})
obj.name = 'bao';
console.log(obj.name)  // bjl
let obj = {};
Object.defineProperty(obj, 'name', {
    value: 'bjl',
    writable: true // 表示可以进行修改
})
obj.name = 'bao';
console.log(obj.name)  // bao
let obj = {
    name: 'bjl'
};
Object.defineProperty(obj, 'name', {
    writable: false  //手动设置name属性不可被修改
})
obj.name = 'bao';
console.log(obj.name)  // bjl

当声明一个对象时,它里面的属性的内部属性的默认值都为 true,也就是说 writable 这时的默认值为 true,这就是为什么上面的 name 如果不想被修改就需要手动去设置 writable 属性的原因。我们可以使用 Object.getOwnPropertyDescriptors() 去检测属性的内部属性的具体描述,如下:

let obj = {
    name: 'bjl'
}    
console.log(Object.getOwnPropertyDescriptors(obj))

Object.defineProperty(obj, 'age', {
    value: 18
})
console.log(Object.getOwnPropertyDescriptors(obj))

打印如下:

 2. 存储描述符(get、set)

当使用 get 或 set 方法,不允许使用 value 和 writable 这两个属性。

  • get:一个给属性提供 getter 的方法,默认值为 undefined
  • set:一个给属性提供 setter 的方法。默认值为 undefined,该方法接受唯一的参数,并将该参数的新值分配给该属性
let obj = {};
let temp = 'bjl';
Object.defineProperty(obj, 'name', {
    get: function() {
        return temp
    },
    set: function(val) {
        temp = val
    }
})
console.log(obj.name)  // bjl
// 当修改属性时,就会触发方法里的set方法
obj.name = 'bao';
console.log(obj.name)  // bao

3. enumerable:表示目标属性是否可以被枚举

有时我们会对对象进行遍历,只有被枚举的属性可以被遍历到,如下所示:

  • for...in...
let obj = {
    name: 'bjl',
    age: 18
}
Object.defineProperty(obj, 'sex', {
    value: '女'
})
for(let key in obj) {
    console.log(key)  // name age
}
  • Object.keys()
let obj = {
    name: 'bjl',
    age: 18
}
Object.defineProperty(obj, 'sex', {
    value: '女'
})
console.log(Object.keys(obj))  // ['name', 'age']
Object.keys(obj).map(key => {
    console.log(key)  // name age
})

我们可以看出,使用 Object.defineProperty() 添加的属性不能被枚举到,也就是不能被遍历到。Object.keys() 返回的是可以被枚举到的数组,所以打印结果如上所述。

添加 enumerable 描述属性可以改变,如下所示:

let obj = {
    name: 'bjl',
    age: 18
}
Object.defineProperty(obj, 'sex', {
    value: '女',
    enumerable: true
})
console.log(Object.keys(obj))  // ['name', 'age', 'sex']
Object.keys(obj).map(key => {
    console.log(key)  // name age sex
})

4. configurable:目标属性是否可以被删除或是否可以再次修改特性

  • 对象原属性默认为 true,可以被删除
let obj = {
    name: 'bjl',
    age: 18
}
delete obj.name;
console.log(obj)  // {age: 18}
  • Object.defineProperty() 内部属性为 false,不可被删除
let obj = {
    name: 'bjl',
    age: 18
}
Object.defineProperty(obj, 'sex', {
    value: '女'
})
delete obj.sex;
console.log(obj)  // {name: 'bjl', age: 18, sex: '女'}
  • 使用 configurable 属性设置可以删除属性
let obj = {
    name: 'bjl',
    age: 18
}
Object.defineProperty(obj, 'sex', {
    value: '女',
    configurable: true
})
Object.defineProperty(obj, 'name', {
    configurable: true
})
delete obj.sex;
delete obj.name;
console.log(obj)  // {age: 18}

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

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

相关文章

C语言自定义类型---进阶

之前的文章中有结构体初阶知识的讲解&#xff0c;对结构体不是很了解的小伙伴可以先去去看一下结构体初阶 结构体&#xff0c;枚举&#xff0c;联合结构体结构体类型的声明特殊的声明结构的自引用结构体变量的定义和初始化结构体内存对齐 <3 <3 <3(重点)那为什么存在内…

Unity记录2.1-动作-多段跳、蹬墙跳、墙体滑落

文章首发及后续更新&#xff1a;https://mwhls.top/4450.html&#xff0c;无图/无目录/格式错误/更多相关请至首发页查看。 新的更新内容请到mwhls.top查看。 欢迎提出任何疑问及批评&#xff0c;非常感谢&#xff01; 汇总&#xff1a;Unity 记录 摘要&#xff1a;实现跳跃、蹬…

若依学习(前后端分离版)——自定义注解@Log(如何自定义注解,实现aop)

如何自定义注解 aop的基本知识与应用 若依对用户的一些更新删除等敏感操作操作进行了日志记录 注解定义和切面处理的项目位置 第一步&#xff1a;自定义注解log 定义了注解的相关信息。这里定义的属性可以在使用时加以定义 注解Target和Retention的作用 第二步切面逻辑…

动手实现一遍Transformer

最近乘着ChatGpt的东风&#xff0c;关于NLP的研究又一次被推上了风口浪尖。在现阶段的NLP的里程碑中&#xff0c;无论如何无法绕过Transformer。《Attention is all you need》成了每个NLP入门者的必读论文。惭愧的是&#xff0c;我虽然使用过很多基于Transformer的模型&#x…

synchronized从入门到踹门

synchronized是什么synchronized是Java关键字&#xff0c;为了维护高并发是出现的原子性问题。技术是把双刃剑&#xff0c;多线程并发给我带来了前所未有的速率&#xff0c;然而在享受快速编程的过程&#xff0c;也给我们带来了原子性问题。如下&#xff1a;public class Main …

【微服务】认识微服务

目录 1.1 单体、分布式、集群 单体 分布式 集群 1.2 系统架构演变 1.2.1 单体应⽤架构 1.2.2 垂直应⽤架构 1.2.3 分布式架构 1.2.4 SOA架构 1.2.5 微服务架构 1.3 微服务架构介绍 微服务架构的常⻅问题 1.4 SpringCloud介绍 1.4.1 SpringBoot和SpringCloud有啥关…

[1.3_2]计算机系统概述——中断和异常

文章目录第一章 计算机系统概述中断和异常&#xff08;一&#xff09;中断的作用&#xff08;二&#xff09;中断的类型&#xff08;三&#xff09;中断机制的基本原理小结第一章 计算机系统概述 中断和异常 中断的作用中断的类型 内中断&#xff08;也称“异常”&#xff09;…

ES之DSL查询文档基础查询

分类 query查询分类 总体规律就是逻辑性的&#xff0c;从外层的你干嘛&#xff0c;到下一层的查询类型&#xff0c;再到下一层的查询字段&#xff08;如果需要的话&#xff09;和然后是查询内容 查询所有 语法 get /索引库名/_serarch {"query":{"查询条件…

【Linux】配置动态IP

动态IP 服务器重启完成之后&#xff0c;我们可以通过linux的指令 ip addr 来查询Linux系统的IP地址&#xff0c;具体信息如 下: 从图中我们可以看到&#xff0c;并没有获取到linux系统的IP地址&#xff0c;这是为什么呢&#xff1f;这是由于启动服务器时未 加载网卡&#x…

2D图像处理:Qt + Opencv使用光度立体法检测Halcon中提供的缺陷图像

文章目录 不需知道光源方向一、光度立体法(后续有时间在查资料研究)1.1 问题1:Slants和Tilts的理解(暂时是理解的)1.2 问题1:Gradient通道数为1,为何像素点对应的值会有两个?1.3 问题2:F(r,c)=(u(r,c),v(r,c)) 关于高斯曲率和平均曲率如何计算的?二、非标定光源实现光…

C++经典20题型,满满知识,看这一篇就够了(含答案)

今天找了20道c的经典题型&#xff0c;看这一篇就够了&#xff0c;全是干货 目录 1、题目&#xff1a;有一对兔子&#xff0c;从出生后第3个月起每个月都生一对兔子&#xff0c;小兔子长到第三个月后每个月又生一对兔子&#xff0c;假如兔子都不死&#xff0c;问每个月的兔子总…

【蓝桥杯集训11】BFS(4 / 4)

目录 844. 走迷宫 - BFS求最短路 1233. 全球变暖 - BFS 845. 八数码 - 最短路BFS 状态表示 一二维坐标转换 为什么BFS保证走的是最短路&#xff1f; 一二维坐标转换&#xff08;nn矩阵&#xff09; 1562.微博转发 - BFS按层遍历 有向图 844. 走迷宫 - BFS求最短路 活…

Centos7安装中文字体

一、背景 最近一直在重写2021年毕设的前端页面&#xff0c;用vue3vite实现的响应式布局&#xff0c;目前完成10%。但在部署到Linux上时&#xff0c;遇到了服务端生成的中文验证码混乱的问题&#xff0c;通过远程断点&#xff0c;排除编码的问题&#xff0c;原来是由于Linux没有…

Linux网络编程 第六天

目录 学习目标 libevent介绍 libevent的安装 libevent库的使用 libevent的使用 libevent的地基-event_base 等待事件产生-循环等待event_loop 使用libevent库的步骤&#xff1a; 事件驱动-event 编写一个基于event实现的tcp服务器&#xff1a; 自带buffer的事件-buff…

深圳大学计软《面向对象的程序设计》实验14 运算符重载2和函数模板

A. 日期比较&#xff08;运算符重载之类型转换&#xff09; 题目描述 定义一个日期类CDate&#xff0c;包含属性&#xff1a;年、月、日&#xff0c;都是整数。 构造函数含单个参数&#xff0c;参数是八位整数&#xff0c;默认值为0&#xff0c;实现整数转为日期类型&#x…

【基于感知损失的无监督泛锐化】

PercepPan: Towards Unsupervised Pan-Sharpening Based on Perceptual Loss &#xff08;PercepPan&#xff1a;基于感知损失的无监督泛锐化&#xff09; 在基于神经网络的全色锐化文献中&#xff0c;作为地面实况标签的高分辨率多光谱图像通常是不可用的。为了解决这个问题…

C++初学笔记整理

目录 1. C关键字 2. 命名空间 1&#xff09;命名空间的引入和概述 2&#xff09;命名空间的定义 3&#xff09;std与命名空间的使用 4).相关特性 3. C输入&输出 4. 缺省参数 1 &#xff09;缺省参数概念 2&#xff09;使用及分类 a.全缺省 b.部分缺省 5. 函数…

力扣-337打家劫舍III(dp)

力扣-337打家劫舍III 1、题目 337. 打家劫舍 III 小偷又发现了一个新的可行窃的地区。这个地区只有一个入口&#xff0c;我们称之为 root 。 除了 root 之外&#xff0c;每栋房子有且只有一个“父“房子与之相连。一番侦察之后&#xff0c;聪明的小偷意识到“这个地方的所有…

【FMCW 01】中频IF信号

FMCW信号 调频连续波(frequency modulated continuous wave&#xff0c;FMCW)顾名思义&#xff0c;就是对信号的频率进行线性调制的信号。 从时域上看&#xff0c;对频率的调制&#xff0c;就像一把连续的锯齿波。其中每一个锯齿叫做一个chirp&#xff0c;其持续的时间叫做ch…

Android仿微信选择图片

效果展示首先先添加用到的权限<uses-permission android:name"android.permission.INTERNET" /><!--获取手机存储卡权限--><uses-permission android:name"android.permission.READ_EXTERNAL_STORAGE"/><uses-permission android:nam…