菜菜: “老大,那个, Object.defineProperty 是什么鬼?”es6
假设咱们有个对象 user ; 咱们要给它增长一个属性 name , 咱们会这么作es5
1
2
3
|
var
user = {};
user.name=
"狂奔的蜗牛"
;
console.log(user);
//{name: "狂奔的蜗牛"}
|
若是想要增长一个sayHi方法叻?spa
1
2
|
user.sayHi=
function
() { console.log(
"Hi !"
) };
console.log(user);
//{name: "狂奔的蜗牛", sayHi: ƒn}
|
Object.defineProperty
就是作这个的.net
那么Object.defineProperty 怎么用?code
Object.defineProperty 须要三个参数(object , propName , descriptor)htm
1 object 对象 => 给谁加
2 propName 属性名 => 要加的属性的名字 【类型:String】
3 descriptor 属性描述 => 加的这个属性有什么样的特性【类型:Object】对象
那么descriptor
这个是个对象 ,他有那些属性呢 ? 别着急咱们一个一个说;ip
既然能够给一个对象增长属性,那么咱们用它来作一下给 user添加 name属性,代码是这样的ci
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:目标属性获取值的方法 原文连接:https://www.jianshu.com/p/6519575e05