前言

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(),该方法包含三个参数,给其添加属性的对象、属性的名称和一个描述符对象。如下:

  1. var person = {};
  2. Object.defineProperty(person, "age", {
  3. enumerable: true,
  4. configurable: false,
  5. value: 18
  6. });
  7. console.log(person.age);// 18
  8. delete person.age
  9. console.log(person.age);// 18
  10.  
  11. Object.defineProperty(person, "height", {
  12. enumerable: false,
  13. value: 50
  14. });
  15. console.log(person);//{age:18,height:50}
  16. for (const key in person) {
  17. console.log(key, person[key]);// age 18
  18. }
  19. Object.defineProperty(person, "name", {
  20. writable: false,
  21. value: "Nicholas"
  22. });
  23. console.log(person.name); // "Nicholas"
  24. person.name = "Greg";
  25. 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() 。如下:

  1. var person = { name: "Nicholas", isAdult: true };
  2. Object.defineProperty(person, 'age', {
  3. get: function () {
  4. return this.age
  5. },
  6. set: function (newValue) {
  7. this.isAdult = newValue >= 18 ? true : false
  8. }
  9. });
  10. console.log(person.isAdult);
  11. person.age = 17
  12. console.log(person.isAdult);

  注意:设置了访问器属性之后不能再设置数据属性,比如设置了 set/get  就不能再设置 writable。

  2、补充:内部属性的操作

  (1)一次性定义多个属性

  ECMAScript 提供了 Object.defineProperties() 方法可以通过多个描述符一次性定义多个属性。它接收两个参数:要为之添加或修改属性的对象和另一个描述符对象,其属性与要添加或修改的属性一一对应。这样所有属性都是同时定义的,并且数据属性的configurable 、 enumerable 和 writable 特性值都是 false。
  1. var person = {}
  2. Object.defineProperties(person, {
  3. 'name': {
  4. value: "Nicholas"
  5. },
  6. 'isAdult': {
  7. writable: true,
  8. configurable: true,
  9. value: true
  10. },
  11. 'age': {
  12. configurable: true,
  13. get() {
  14. return this.age
  15. },
  16. set(newValue) {
  17. this.isAdult = newValue >= 18 ? true : false
  18. }
  19. }
  20. })
  21. console.log(person.isAdult);// true
  22. person.age = 17// 首先到age的set方法,然后访问isAdult的writable属性并修改value值
  23. console.log(person.isAdult);// false

  (2)读取属性的特性   Object.getOwnPropertyDescriptor()获取传入对象给定属性的描述符。Object.getOwnPropertyDescriptors()获取给定对象的全部属性描述符。

  1. var person = { name: "Nicholas", isAdult: true };
  2. Object.defineProperty(person, 'age', {
  3. get: function () {
  4. return this.age
  5. },
  6. set: function (newValue) {
  7. this.isAdult = newValue >= 18 ? true : false
  8. }
  9. });
  10. var descriptor = Object.getOwnPropertyDescriptor(person, 'age')
  11. console.log(descriptor);
  12. //configurable: false
  13. // enumerable: false
  14. // get: ƒ ()
  15. // set: ƒ (newValue)
  16. console.log(Object.getOwnPropertyDescriptors(person));
  17. // {name:{...},isAdult:{...},age:{...}}

  3、对象内部特性和 Object.defineProperty() 应用

  (1)手写实现 const

  1. function myConst(key, value) {
  2. window.key = value // 把要定义的key挂载到window下,并赋值value
  3. Object.defineProperty(window, key, {
  4. enumerable: false,
  5. configurable: false,
  6. get: function () {
  7. return value
  8. },
  9. set: function (data) {
  10. if (data !== value) { // 当要对当前属性进行赋值时,则抛出错误!
  11. throw new TypeError('Assignment to constant variable.')
  12. } else {
  13. return value
  14. }
  15. }
  16. })
  17. }
  18. myConst('a', 2)
  19. console.log(a);
  20. a = 3//报错 :Assignment to constant variable.

  (2)手动实现每次访问一个属性时,值加一,使得a==1&&a==2&a==3成为可能

  1. window.b = 0
  2. Object.defineProperty(window, "a", {
  3. get() {
  4. this.b++
  5. return this.b
  6. },
  7. set(value) {
  8. this.b = value
  9. }
  10. })
  11. if (a == 1 && a == 2 & a == 3) {
  12. console.log("111");
  13. }//111

写在最后

  以上就是本文的全部内容,希望给读者带来些许的帮助和进步,方便的话点个关注,小白的成长之路会持续更新一些工作中常见的问题和技术点。

js--对象内部属性与 Object.defineProperty()的更多相关文章

  1. js对象的属性:数据(data)属性和访问器(accessor)属性

    此文为转载,原文: 深入理解对象的数据属性与访问器属性 创建对象的方式有两种:第一种,通过new操作符后面跟Object构造函数,第二种,对象字面量方式.如下 var person = new Obj ...

  2. C#对象内部属性排序测试

    构建对象: class SortGrid { int indexI; int indexJ; public SortGrid(int x, int y) { indexI = x; indexJ = ...

  3. js对象中属性调用.和[] 两种方式的区别

    JS 调用属性一般有两种方法——点和中括号的方法. 标准格式是对象.属性(不带双引号),注意一点的是:js对象的属性,key标准是不用加引号的,加也可以,特别的情况必须加,如果key数字啊,表达式啊等 ...

  4. 对象内部属性[[Class]]

    1.概述 所有的typeof返回值为‘object’的对象都包含一个内部属性[[Class]],我们将它可以看做内部的分类,而非传统面向对象意义的分类.这个属性无法直接访问,一般通过Object.pr ...

  5. JS对象中属性的增删改查

    对象属于一种复合的数据类型,在对象中可以保存多个不同数据类型的属性 对象的分类:           1.内建对象                 -在ES标准中定义的对象,在任何的ES的实现中都可以 ...

  6. js基础学习之-js对象的属性

    Js属性 1. 设置属性 1)  对象. 2)  对象[‘属性名’] 3)   GetAttribute函数 2. 获取属性 1)  变量=对象. 2)  变量=对象[‘属性名’] 3)  GetAt ...

  7. js对象的属性问题

    ES6之前js的对象的属性只能是字符串, <html> <head> <script type="text/javascript"> var a ...

  8. JS对象或属性的不变性

    提到不变性,不得不提一个概念: 对象常量定义:结合可写性与可配置性可以创建一个真正的常量属性(不可修改.重定义.删除) 不变性可划分为以下几个等级: 1)禁止扩展:Object.preventExte ...

  9. js对象常用属性和方法:复制一个对象,获取一个对象的所有key和所有value的方法

    记录对象的一些实用使用方法及属性 // Object.assign() 多个对象合并 key相同则后面的覆盖前面的 const target = { a: 1, b: 2 }; const sourc ...

随机推荐

  1. [bzoj5508]甲苯先生的字符串

    首先定义状态f[i][j]表示长度为i的串以j为结尾有多少符合条件的串,发现$f[i][j]=\sum f[i-1][k]$(j和k可以相邻),这个用矩阵乘法优化一下即可. 1 #include< ...

  2. linux中为何每次修改完配置文件后都需要重新加载配置文件

    1.大家刚接触linux时,可能会有这样的疑问:为什么每次修改完配置文件之后,总是要重新加载配置文件才能生效?或者需要重启后才能生效?   之前听过一个解释是这样子的:   "修改了文件内容 ...

  3. redis序列化和反序列化的操作-(以前咋操作我都忘记了)

    //拿到数据,redis如果有则将现在有的传进去,如果没有则获取接口 ExWritPropertyVo ExWritPropertyVo = new ExWritPropertyVo(); ExWri ...

  4. 31、下一个排列 | 算法(leetode,附思维导图 + 全部解法)300题

    零 标题:算法(leetode,附思维导图 + 全部解法)300题之(31)下一个排列 一 题目描述 二 解法总览(思维导图) 三 全部解法 1 方案1 1)代码: // 方案1 "双指针法 ...

  5. Terminator--最强Ubuntu终端

    个人使用的一个非常好用的Ubuntu终端软件. Terminator的安装和配置 安装 Ubuntu sudo add-apt-repository ppa:gnome-terminator sudo ...

  6. dart系列之:时间你慢点走,我要在dart中抓住你

    目录 简介 DateTime Duration 总结 简介 时间和日期是我们经常会在程序中使用到的对象.但是对时间和日期的处理因为有不同时区的原因,所以一直以来都不是很好用.就像在java中,为时间和 ...

  7. AGC050B Three Coins

    做的时候有思考到是否能转化成移动点问题,但是没有清晰的把他解释出来. NOIP的时候也一样,T3也有考虑到是否能转为差分,但是也没有清晰的写出来. 自己做题的时候应尽量保证草稿纸和思绪的清晰,而不是在 ...

  8. 标准非STL容器 : bitset

    1. 概念 什么是"标准非STL容器"?标准非STL容器是指"可以认为它们是容器,但是他们并不满足STL容器的所有要求".前文提到的容器适配器stack.que ...

  9. C语言中内存对齐与结构体

    结构体 结构体是一种新的数据类型,对C语言的数据类型进行了极大的扩充. struct STU{ int age; char name[15]; }; struct STU a; //结构体实例 str ...

  10. 点击下拉选择触发事件【c#】

    <asp:DropDownList ID="ddlRegionList" runat="server" AutoPostBack="true&q ...