js--对象内部属性与 Object.defineProperty()
前言
JavaScript 中允许使用一些内部特性来描述属性的特征,本文来总结一下对象内部属性与 Object.defineProperty() 的相关知识。
正文
1、属性类型
js中使用某些内部属性来描述属性的特征,比如描述属性是否可以枚举,是否可以修改等特征,我们无法访直接问属性的这些特征,但是可以通过[[]]的方式来将某个特性标识为内部属性。这些内部属性分为数据属性和访问器属性。
(1)数据属性
数据属性包含一个保存数据值的位置。值会从这个位置读取,也会写入到这个位置。数据属性有 4个特性描述它们的行为。
a、[[Configurable]]表示能否通过 delete 删除属性从而重新定义属性,能否修改属性的特性,或者能否把数据属性修改为访问器属性,默认 true。
b、[[Enumerable]]表示能否通过 for-in 循环遍历出属性 默认 true。
c、[[Writable]]表示能否修改属性的值,默认 true,设置为 false 指改属性只读。
d、[[Value]]保存这个属性的数据值,读写属性的值都是在这个位置,默认值为 undefined。
要修改对象默认数据属性,必须使用es5的 Object.defineProperty(),该方法包含三个参数,给其添加属性的对象、属性的名称和一个描述符对象。如下:
var person = {};
Object.defineProperty(person, "age", {
enumerable: true,
configurable: false,
value: 18
});
console.log(person.age);// 18
delete person.age
console.log(person.age);// 18
Object.defineProperty(person, "height", {
enumerable: false,
value: 50
});
console.log(person);//{age:18,height:50}
for (const key in person) {
console.log(key, person[key]);// age 18
}
Object.defineProperty(person, "name", {
writable: false,
value: "Nicholas"
});
console.log(person.name); // "Nicholas"
person.name = "Greg";
console.log(person.name); // "Nicholas"
注意:在调用 Object.defineProperty() 时, configurable 、 enumerable 和 writable 的值如果不指定,则都默认为 false 。
(2)访问器属性
访问器属性 ==访问器属性不包含数据值,他们包含setter和getter函数,这两个函数并不是必须的
a、[[Configurable]]表示能否通过delete删除属性从而重新定义属性,能否修改属性的特性,或者能否把数据属性修改为访问器属性,默认 true。
b、[[Enumerable]]表示能否通过for-in循环遍历出属性 默认 true。
c、[[Set]]写入属性的函数,默认 undefined。
d、[[Get]]读取属性的函数,默认 undefined。
同样,访问器属性是不能直接定义的,必须使用 Object.defineProperty() 。如下:
var person = { name: "Nicholas", isAdult: true };
Object.defineProperty(person, 'age', {
get: function () {
return this.age
},
set: function (newValue) {
this.isAdult = newValue >= 18 ? true : false
}
});
console.log(person.isAdult);
person.age = 17
console.log(person.isAdult);
注意:设置了访问器属性之后不能再设置数据属性,比如设置了 set/get 就不能再设置 writable。
2、补充:内部属性的操作
(1)一次性定义多个属性
var person = {}
Object.defineProperties(person, {
'name': {
value: "Nicholas"
},
'isAdult': {
writable: true,
configurable: true,
value: true
},
'age': {
configurable: true,
get() {
return this.age
},
set(newValue) {
this.isAdult = newValue >= 18 ? true : false
}
}
})
console.log(person.isAdult);// true
person.age = 17// 首先到age的set方法,然后访问isAdult的writable属性并修改value值
console.log(person.isAdult);// false
(2)读取属性的特性 Object.getOwnPropertyDescriptor()获取传入对象给定属性的描述符。Object.getOwnPropertyDescriptors()获取给定对象的全部属性描述符。
var person = { name: "Nicholas", isAdult: true };
Object.defineProperty(person, 'age', {
get: function () {
return this.age
},
set: function (newValue) {
this.isAdult = newValue >= 18 ? true : false
}
});
var descriptor = Object.getOwnPropertyDescriptor(person, 'age')
console.log(descriptor);
//configurable: false
// enumerable: false
// get: ƒ ()
// set: ƒ (newValue)
console.log(Object.getOwnPropertyDescriptors(person));
// {name:{...},isAdult:{...},age:{...}}
3、对象内部特性和 Object.defineProperty() 应用
(1)手写实现 const
function myConst(key, value) {
window.key = value // 把要定义的key挂载到window下,并赋值value
Object.defineProperty(window, key, {
enumerable: false,
configurable: false,
get: function () {
return value
},
set: function (data) {
if (data !== value) { // 当要对当前属性进行赋值时,则抛出错误!
throw new TypeError('Assignment to constant variable.')
} else {
return value
}
}
})
}
myConst('a', 2)
console.log(a);
a = 3//报错 :Assignment to constant variable.
(2)手动实现每次访问一个属性时,值加一,使得a==1&&a==2&a==3成为可能
window.b = 0
Object.defineProperty(window, "a", {
get() {
this.b++
return this.b
},
set(value) {
this.b = value
}
})
if (a == 1 && a == 2 & a == 3) {
console.log("111");
}//111
写在最后
以上就是本文的全部内容,希望给读者带来些许的帮助和进步,方便的话点个关注,小白的成长之路会持续更新一些工作中常见的问题和技术点。

js--对象内部属性与 Object.defineProperty()的更多相关文章
- js对象的属性:数据(data)属性和访问器(accessor)属性
此文为转载,原文: 深入理解对象的数据属性与访问器属性 创建对象的方式有两种:第一种,通过new操作符后面跟Object构造函数,第二种,对象字面量方式.如下 var person = new Obj ...
- C#对象内部属性排序测试
构建对象: class SortGrid { int indexI; int indexJ; public SortGrid(int x, int y) { indexI = x; indexJ = ...
- js对象中属性调用.和[] 两种方式的区别
JS 调用属性一般有两种方法——点和中括号的方法. 标准格式是对象.属性(不带双引号),注意一点的是:js对象的属性,key标准是不用加引号的,加也可以,特别的情况必须加,如果key数字啊,表达式啊等 ...
- 对象内部属性[[Class]]
1.概述 所有的typeof返回值为‘object’的对象都包含一个内部属性[[Class]],我们将它可以看做内部的分类,而非传统面向对象意义的分类.这个属性无法直接访问,一般通过Object.pr ...
- JS对象中属性的增删改查
对象属于一种复合的数据类型,在对象中可以保存多个不同数据类型的属性 对象的分类: 1.内建对象 -在ES标准中定义的对象,在任何的ES的实现中都可以 ...
- js基础学习之-js对象的属性
Js属性 1. 设置属性 1) 对象. 2) 对象[‘属性名’] 3) GetAttribute函数 2. 获取属性 1) 变量=对象. 2) 变量=对象[‘属性名’] 3) GetAt ...
- js对象的属性问题
ES6之前js的对象的属性只能是字符串, <html> <head> <script type="text/javascript"> var a ...
- JS对象或属性的不变性
提到不变性,不得不提一个概念: 对象常量定义:结合可写性与可配置性可以创建一个真正的常量属性(不可修改.重定义.删除) 不变性可划分为以下几个等级: 1)禁止扩展:Object.preventExte ...
- js对象常用属性和方法:复制一个对象,获取一个对象的所有key和所有value的方法
记录对象的一些实用使用方法及属性 // Object.assign() 多个对象合并 key相同则后面的覆盖前面的 const target = { a: 1, b: 2 }; const sourc ...
随机推荐
- Codeforces 1365G - Secure Password(思维题)
Codeforces 题面传送门 & 洛谷题面传送门 首先考虑一个询问 \(20\) 次的方案,考虑每一位,一遍询问求出下标的这一位上为 \(0\) 的位置上值的 bitwise or,再一遍 ...
- 洛谷 P3676 - 小清新数据结构题(动态点分治)
洛谷题面传送门 题目名称好评(实在是太清新了呢) 首先考虑探究这个"换根操作"有什么性质.我们考虑在换根前后虽然每个点的子树会变,但整棵树的形态不会边,换句话说,割掉每条边后,得到 ...
- 洛谷 P3688 - [ZJOI2017]树状数组(二维线段树+标记永久化)
题面传送门 首先学过树状数组的应该都知道,将树状数组方向写反等价于前缀和 \(\to\) 后缀和,因此题目中伪代码的区间求和实质上是 \(sum[l-1...n]-sum[r...n]=sum[l-1 ...
- DirectX12 3D 游戏开发与实战第七章内容(上)
利用Direct3D绘制几何体(续) 学习目标 学会一种无须每帧都要刷新命令队列的渲染流程,以此来优化性能 了解另外两种根签名参数类型:根常量和根描述符 探索如何在程序中生成和绘制常见的几何体:如栅格 ...
- 【Python小试】计算蛋白序列中指定氨基酸所占的比例
编码 from __future__ import division def get_aa_percentage(protein, aa_list=['A','I','L','M','F','W',' ...
- 汽车C2M模式综述
- 从零开始学习oracle
引用博客:https://blog.csdn.net/qq_36998053/article/details/82725765 )Oracle之<环境配置> (二)Oracle之<基 ...
- (转载) IBM DB2数据库odbc配置步骤详解
[IT168 技术] 首先安装IBM DB2 odbc driver 1):可以单独下载DB2 Run-Time Client,大约(86.6m),安装后则odbc驱动程序安装成功.下载地址:ftp: ...
- 到底什么是自动化优先思维?与RPA有什么关系?
基于RPA的自动化优先,正在成为广大组织的主流管理思维 到底什么是自动化优先思维?与RPA有什么关系? 如何用RPA简单快速的打造一个自动化优先的组织? 文/王吉伟 在IT运维项目中,组织经常会遇到先 ...
- 基于MQTT协议实现远程控制的"智能"车
智能,但不完全智能 虽然我不觉得这玩意儿有啥智能的,但都这么叫就跟着叫喽. 时隔好几天才写的 其实在写这篇博文的时候我已经在做升级了,并且已经到了中后期阶段了. 主要是业余时间做着玩,看时间了. 规格 ...
