js中Object.defineProperty()方法的解释
菜菜: “老大,那个, 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=false
的birth
属性并没有被遍历出来,遍历 => 其实就是枚举(个人理解啦,不喜勿喷哦~)
总结 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.除了可以给新定义的属性设置特性,也可以给已有的属性设置特性哈
)
最后我们来说说,最重要的两个属性 set
和get
(即存取器描述:定义属性如何被存取),这两个属性是做什么用的呢?我们通过代码来看看
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()方法的解释的更多相关文章
- js中Object.defineProperty()和defineProperties()
在介绍js中Object.defineProperty()和defineProperties()之前,我们了解下js中对象两种属性的类型:数据属性和访问器属性. 数据属性 数据属性包含一个数据的位置, ...
- vue2.x版本中Object.defineProperty对象属性监听和关联
前言 在vue2.x版本官方文档中 深入响应式原理 https://cn.vuejs.org/v2/guide/reactivity.html一文的解释当中,Object.defineProperty ...
- Vue el与data的两种写法 && Object.defineProperty方法
1 <!DOCTYPE html> 2 <html> 3 <head> 4 <meta charset="UTF-8" /> 5 & ...
- Object.defineProperties()和Object.defineProperty()方法
Object.defineProperty() 方法会直接在一个对象上定义一个新属性,或者修改一个对象的现有属性, 并返回这个对象. 语法:Object.defineProperty(obj, pro ...
- jQuery与JS中的map()方法使用
1.jquery中的map()方法 首先看一个简单的实例: $("p").append( $("input").map(function(){ return $ ...
- ES6中Object.assign() 方法
ES6中Object.assign() 方法 1. 对象合并Object.assign 方法用于对象的合并,将源对象(source)的所有可枚举属性,复制到目标对象上.如下代码演示: var targ ...
- Object.defineProperty方法 使用
Object.defineProperty() 方法会直接在一个对象上定义一个新属性,或者修改一个对象的现有属性, 并返回这个对象. 语法: Object.defineProperty(obj, pr ...
- Object.defineProperty() 方法会直接在一个对象上定义一个新属性,或者修改一个已经存在的属性, 并返回这个对象。
Object.defineProperty() 方法会直接在一个对象上定义一个新属性,或者修改一个已经存在的属性, 并返回这个对象. 语法EDIT Object.defineProperty(obj, ...
- JavaScript Object.defineProperty()方法详解
Object.defineProperty() 方法直接在一个对象上定义一个新属性,或者修改一个已经存在的属性, 并返回这个对象. 语法 Object.defineProperty(obj, prop ...
随机推荐
- python学习第一天变量命名规范和变量作用
变量的命名 python中的变量跟其他编程语言变量一样 1,由字母,下划线,数字组成 2,数字不能做变量名开头 3,变量名尽量有意义和短,,也可以驼峰,不要很low ,比如说是 中文,变量名很长 py ...
- CodeForces 711D Directed Roads (DFS找环+组合数)
<题目链接> 题目大意: 给定一个$n$条边,$n$个点的图,每个点只有一条出边(初始状态),现在能够任意对图上的边进行翻转,问你能够使得该有向图不出先环的方案数有多少种. 解题分析: 很 ...
- 远程连接不上centos的mysql的解决方法
1.防火墙没有开放3306端口 centos 有两种防火墙 FirewallD和iptables防火墙 centos7 使用的是FirewallD防火墙. 1.FirewallD防火墙开放3306端口 ...
- 使用easyui框架 设置时间格式
之前做的一个商城项目,后台系统页面使用Easyui做的,记录一个当时卡住的地方: 首先通过<table>标记创建数据网格(datagrid) <table class="e ...
- ES6——解构赋值
解构赋值: 注意: 1.左右两边结构必须一样 练习1,2,3 2.右边必须是个东西(有值)练习4 3.声明和赋值不能分开(必须在一句话里完成)练习5 /* 练习1: // let arr = [1,2 ...
- 2018-2-13-C#-搜索算法
title author date CreateTime categories C# 搜索算法 lindexi 2018-2-13 17:23:3 +0800 2018-2-13 17:23:3 +0 ...
- 系统INIT 启动脚本的结构/etc/rc.d/init.d/*
- smbsh - 允许用UNIX命令访问NT文件系统
总览 smbsh 描述 此程序是Samba套件的一部分. smbsh允许你用UNIX命令诸如ls,egrep和rcp等来访问NT文件系统.必须用动态链接的shell以便使smbsh工作正常. 从命令提 ...
- go语言从例子开始之Example32.打点器
定时器 是当你想要在未来某一刻执行一次时使用的 - 打点器 则是当你想要在固定的时间间隔重复执行准备的.这里是一个打点器的例子,它将定时的执行,直到我们将它停止. Example: package m ...
- 转载:PhpExcel使用方法
下面是总结的几个使用方法 include 'PHPExcel.php'; include 'PHPExcel/Writer/Excel2007.php'; //或者include 'PHPExcel/ ...