(在园里看到一篇讲解Object.defineProperty()的非常好的博文,我把他转载过来,给自己留个笔记。)

原文出处https://www.cnblogs.com/junjun-001/p/11761252.html#commentform

菜菜: “老大,那个, Object.defineProperty 是什么鬼?”

假设我们有个对象 user ; 我们要给它增加一个属性 name , 我们会这么做

1

2

3

var user = {};

user.name="狂奔的蜗牛";

console.log(user);//{name: "狂奔的蜗牛"}

如果想要增加一个sayHi方法叻?

1

2

user.sayHi=function () { console.log("Hi !") };

console.log(user);//{name: "狂奔的蜗牛", sayHi: ƒn}

Object.defineProperty 就是做这个的

那么Object.defineProperty 怎么用?

Object.defineProperty 需要三个参数(object , propName , descriptor)

1 object 对象 => 给谁加
2 propName 属性名 => 要加的属性的名字 【类型:String】
3 descriptor 属性描述 => 加的这个属性有什么样的特性【类型:Object】

那么descriptor这个是个对象 ,他有那些属性呢 ? 别着急我们一个一个说;

既然可以给一个对象增加属性,那么我们用它来做一下给 user添加 name属性,代码是这样的

1

2

3

4

5

var user = {};

Object.defineProperty(user,"name",{

 value:"狂奔的蜗牛"

})

console.log(user);//{name: "狂奔的蜗牛"}

说明 是的还是那个经典的value属性,他就是设置属性值的。

等等,属性值只能为字符串吗?我们的 number function Object boolean 等呢?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

var user = {};

Object.defineProperty(user,"name",{

 value:"狂奔的蜗牛"

})

Object.defineProperty(user,"isSlow",{

 value:true

})

Object.defineProperty(user,"sayHi",{

 value:function () { console.log("Hi !") }

})

Object.defineProperty(user,"age",{

 value:12

})

Object.defineProperty(user,"birth",{

 value:{

  date:"2018-06-29",

  hour:"15:30"

 }

})

console.log(user);

说明 事实证明任何类型的数据都是可以的哦~

问题又来了,如果 user对象已经有了name属性,我们可以通过Object.defineProperty改变这个值吗?

我们来试试

1

2

3

4

5

6

7

var user = {};

Object.defineProperty(user,"name",{

 value:"狂奔的蜗牛"

})

console.log(user);

user.name="新=>狂奔的蜗牛"

console.log(user);

咦??为什么我改了没作用勒??

原因:上边说了descriptor有很多属性,除了value属性还有个 writable【顾名思义属性是否可以被重新赋值】接受数据类型为 boolean(默认为false) true => 支持被重新赋值 false=>只读

哦哦,原来如果我没设置writable值的时候就默认只读啊,所以才改不掉

那我们看看,设置为true,是不是就可以改掉了。

1

2

3

4

5

6

7

8

var user = {};

Object.defineProperty(user,"name",{

 value:"狂奔的蜗牛",

 writable:true

})

console.log(user);

user.name="新=>狂奔的蜗牛"

console.log(user);

这个descriptor还有其他的属性吗?enumerable【顾名思义属性是否可以被枚举】接受数据类型为 boolean(默认为false) true => 支持被枚举 false=>不支持

额。。。枚举??什....什么意思?

假设我们想知道这个 user对象有哪些属性我们一般会这么做

1

2

3

4

5

6

7

8

9

10

11

12

13

14

var user ={

 name:"狂奔的蜗牛",

 age:25

} ;

//es6

var keys=Object.keys(user)

console.log(keys);// ['name','age']

//es5

var keys=[];

for(key in user){

 keys.push(key);

}

console.log(keys);// ['name','age']

如果我们使用 Object.的方式定义属性会发生什么呢?我们来看下输出

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

var user ={

 name:"狂奔的蜗牛",

 age:25

} ;

//定义一个性别 可以被枚举

Object.defineProperty(user,"gender",{

 value:"男",

 enumerable:true

})

//定义一个出生日期 不可以被枚举

Object.defineProperty(user,"birth",{

 value:"1956-05-03",

 enumerable:false

})

//es6

var keys=Object.keys(user)

console.log(keys);

// ["name", "age", "gender"]

console.log(user);

// {name: "狂奔的蜗牛", age: 25, gender: "男", birth: "1956-05-03"}

console.log(user.birth);

// 1956-05-03

说明 很明显,我们定义为 enumerable=falsebirth属性并没有被遍历出来,遍历 => 其实就是枚举(个人理解啦,不喜勿喷哦~)

总结 enumerable 属性取值为 布尔类型 true | false 默认值为 false,为真属性可以被枚举;反之则不能。此设置不影响属性的调用和 查看对象的值。

configurable 是接下来我们要讲的一个属性,这个属性有两个作用:

1 属性是否可以被删除
2 属性的特性在第一次设置之后可否被重新定义特性

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

var user ={

 name:"狂奔的蜗牛",

 age:25

} ;

//定义一个性别 不可以被删除和重新定义特性

Object.defineProperty(user,"gender",{

 value:"男",

 enumerable:true,

 configurable:false

})

//删除一下

delete user.gender;

console.log(user);//{name: "狂奔的蜗牛", age: 25, gender: "男"}

//重新定义特性

Object.defineProperty(user,"gender",{

 value:"男",

 enumerable:true,

 configurable:true

})

// Uncaught TypeError: Cannot redefine property: gender

//会报错,如下图

设置为 true

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

var user ={

 name:"狂奔的蜗牛",

 age:25

} ;

//定义一个性别 可以被删除和重新定义特性

Object.defineProperty(user,"gender",{

 value:"男",

 enumerable:true,

 configurable:true

})

//删除前

console.log(user);

// {name: "狂奔的蜗牛", age: 25, gender: "男"}

//删除一下

delete user.gender;

console.log(user);

// {name: "狂奔的蜗牛", age: 25}

//重新定义特性

Object.defineProperty(user,"gender",{

 value:"男",

 enumerable:true,

 configurable:false

})

//删除前

console.log(user);

// {name: "狂奔的蜗牛", age: 25, gender: "男"}

//删除一下 删除失败

delete user.gender;

console.log(user);

// {name: "狂奔的蜗牛", age: 25, gender: "男"}

总结 configurable设置为 true 则该属性可以被删除和重新定义特性;反之属性是不可以被删除和重新定义特性的,默认值为false(Ps.除了可以给新定义的属性设置特性,也可以给已有的属性设置特性哈

最后我们来说说,最重要的两个属性 setget(即存取器描述:定义属性如何被存取),这两个属性是做什么用的呢?我们通过代码来看看

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

var user ={

 name:"狂奔的蜗牛"

} ;

var count = 12;

//定义一个age 获取值时返回定义好的变量count

Object.defineProperty(user,"age",{

 get:function(){

  return count;

 }

})

console.log(user.age);//12

//如果我每次获取的时候返回count+1呢

var user ={

 name:"狂奔的蜗牛"

} ;

var count = 12;

//定义一个age 获取值时返回定义好的变量count

Object.defineProperty(user,"age",{

 get:function(){

  return count+1;

 }

})

console.log(user.age);//13

接下来我不用解释了吧,你想在获取该属性的时候对值做什么随你咯~

来来来,我们看看 set,不多说上代码

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

var user ={

 name:"狂奔的蜗牛"

} ;

var count = 12;

//定义一个age 获取值时返回定义好的变量count

Object.defineProperty(user,"age",{

 get:function(){

  return count;

 },

 set:function(newVal){

  count=newVal;

 }

})

console.log(user.age);//12

user.age=145;

console.log(user.age);//145

console.log(count);//145

//等等,如果我想设置的时候是 自动加1呢?我设置145 实际上设置是146

var user ={

 name:"狂奔的蜗牛"

} ;

var count = 12;

//定义一个age 获取值时返回定义好的变量count

Object.defineProperty(user,"age",{

 get:function(){

  return count;

 },

 set:function(newVal){

  count=newVal+1;

 }

})

console.log(user.age);//12

user.age=145;

console.log(user.age);//146

console.log(count);//146

说明 注意:当使用了getter或setter方法,不允许使用writable和value这两个属性(如果使用,会直接报错滴)

get 是获取值的时候的方法,类型为 function ,获取值的时候会被调用,不设置时为 undefined

set 是设置值的时候的方法,类型为 function ,设置值的时候会被调用,undefined

get或set不是必须成对出现,任写其一就可以

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

var user ={

 name:"狂奔的蜗牛"

} ;

var count = 12;

//定义一个age 获取值时返回定义好的变量count

Object.defineProperty(user,"age",{

 get:function(){

  console.log("这个人来获取值了!!");

  return count;

 },

 set:function(newVal){

  console.log("这个人来设置值了!!");

  count=newVal+1;

 }

})

console.log(user.age);//12

user.age=145;

console.log(user.age);//146

【完结】

Object.defineProperty方法直接在一个对象上定义一个新属性,或者修改一个已经存在的属性, 并返回这个对象

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

js中Object.defineProperty()方法的解释相关推荐

  1. JS中Object的方法汇总,包括assign、create、prototype等等

    JavaScript Object JS的Object到底是啥东西呢?它有啥东西呢? 我们简单知道的,就是new一个Object实例对象,那这个实例对象又何Object又有什么关系呢? 先打印一下看看 ...

  2. php object keys_原生js中Object.keys方法详解

    实际开发中,有时需要知道对象的所有属性,原生js提供了一个方法Object.keys(). Object.keys(obj)返回的是一个数组,该数组的所有元素都是字符串.这些元素是来自于给定的obj可 ...

  3. JS中Object.entries()方法

    Object.entries()方法返回一个给定对象自身可枚举属性的键值对数组,其排列与使用 for-in 循环遍历该对象时返回的顺序一致(区别在于 for-in 循环还会枚举原型链中的属性). co ...

  4. js中 Object.freeze 方法

    Object.freeze() Object.freeze() 方法可以冻结一个对象,冻结指的是不能向这个对象添加新的属性,不能修改其已有属性的值,不能删除已有属性,以及不能修改该对象已有属性的可枚举 ...

  5. Js中Object方法

    1.Object.assign() Object.assign()是通过复制一个或多个对象,创建一个新对象. var source1 = { a: 1 }; var source2 = { b: 2 ...

  6. js之Object的方法理解

    我们都知道在js里面一切都是对象,window也是个对象,对象里拥有很多属性和方法,可以了解一下对象. 在控制台打印一下Object对象,console.dir(Object),我们可以看到Objec ...

  7. 对javscript中Object.defineProperty的理解

      自己在使用vue的过程中经常会用到听到数据双向绑定这个词,而且我们还可以直接通过调用this.msg(this表示vue实例),来获取data上的数据,以前一直不太明白为什么可以这样获取,直到有一 ...

  8. Object.defineProperty方法(详解)

    OK,这一篇主要想说一下Object.defineProperty这个方法. 这个方法也是Vue数据双向绑定原理的常见面试题 所以也是有必要好好掌握的哦 首先我们知道JS中是支持面向对象编程的,也是有 ...

  9. java的object有show_Java中 Object的方法

    Java中 Object的方法 构造方法摘要 Object() 方法摘要 protected Object clone() 创建并返回此对象的一个副本. boolean equals(Object o ...

  10. JavaScript Object.defineProperty()方法详解

    Object.defineProperty() 方法直接在一个对象上定义一个新属性,或者修改一个已经存在的属性,并返回这个对象.因此,又称为属性拦截器.在前端中,webpack以及vue的原理都应用了 ...

最新文章

  1. 安卓机高的地图要用浏览器打开_浏览器F12操作概述
  2. openpyxl 读写 excel
  3. Reactor模式和Proactor模式
  4. nssstring 转换大小写
  5. OpenCV-寻找非零点cv::findNonZero
  6. mysql-proxy读写分离,负载均衡
  7. windows ping 端口测试
  8. 【可视化】盒须图 散点图 柱状图 折线图 饼图
  9. linux清理垃圾缓存文件在哪,Linux系统需要清理垃圾文件和优化系统吗?
  10. 类型 List 不是通用的;不能使用参数() 将它参数化
  11. 篆刻小站之设计与开发
  12. 联想拯救者Y9000P 2022 配置
  13. 4342. 就一勾子 HDU1698 , kuangbin专题
  14. php kestrel,转载 kestrel php 讯息队列
  15. EOS系列六:wallet钱包、key公私钥对、account帐号的关系
  16. STM32CubeIDE编译时出现的问题汇总
  17. python获取IP地址方法
  18. 四款主流杀毒软件病毒库的备份
  19. Python爬虫_02_Python基础知识整理
  20. Kodein在ktor中的使用

热门文章

  1. 【JAVAWEB复习系列】第二部分
  2. 基于Andriod的简易记事本APP设计与实现
  3. Python基础知识day2
  4. 别把职场当官斗,聪明人都在自我成长
  5. 22.Consent 确认逻辑实现
  6. Python之使用代理服务器访问网页
  7. 页错误异常处理(page fault)的实现
  8. 慕思618静悄悄,暴利生意做不下去了?
  9. linux读取文件头错误,Linux系统grub常见错误问题解决
  10. 图片处理工具类ImageHelper