Object.defineProperty
- 看了几篇网上相关博客,发现存在很多问题,有很多误导性,特此梳理总结一下
理解数据劫持
- 数据劫持,其实就是数据代理。
- 数据劫持,指的是在访问或者修改对象的某个属性时,通过一段代码拦截这个行为,进行额外的操作或者修改返回结果。
了解Object.defineProperty() 方法
- Object.defineProperty() 方法会直接在一个对象上定义一个新属性,或者修改一个对象的现有属性,并返回此对象。
Object.defineProperty(obj, prop, descriptor)
- obj 要定义属性的对象。
- prop 要定义或修改的属性的名称或 Symbol 。
- descriptor 要定义或修改的属性描述符。
- 属性描述符有两种主要形式:数据描述符和存取描述符。
- 数据描述符是一个具有值的属性,该值可以是可写的,也可以是不可写的。有
configurable 、enumerable 、value 、writable 四个配置属性。 - 存取描述符是由 getter 函数和 setter 函数所描述的属性。有
configurable 、enumerable 、get 、set 四个配置属性。 - 一个描述符只能是这两者其中之一;不能同时是两者。
当使用了存取描述符get或set方法,不允许使用writable和value这两个属性,同理当使用了数据描述符writable和value,不允许使用get或set方法。
添加对象属性
通过数据描述符创建
const person = {};
Object.defineProperty(person , 'name', {
value: '小陈'
})
console.log(person.name)
通过存取描述符修改
const person = {};
Object.defineProperty(person ,'name',{
get:function(){
console.log('触发get');
return '小陈'
},
})
console.log(person.name)
修改对象属性值
通过数据描述符修改
writable
const person = {
name: '小陈'
};
Object.defineProperty(person, 'name', {
writable: true,
})
console.log(person.name)
person.name = '小王';
console.log(person.name)
当不配置writable时,其默认值是false,属性的值(value)是不能被赋值运算符改变; 当配置writable为true时,属性的值(value)才能被运算符改变; 所以再给name属性值重新赋值,并不会改变name的值。
通过存取描述符修改
const person = {
name: '小陈'
};
let nameValue = person.name
Object.defineProperty(person, 'name', {
configurable: true,
get: function() {
console.log('触发get');
return nameValue
},
set: function(newValue) {
console.log('触发set')
nameValue = newValue
}
})
console.log(person.name)
person.name = '小王';
console.log(person.name)
配置对象属性
- configurable 、enumerable配置对数据描述符和存取描述符设置效果一致。
configurable
- 表示对象的属性是否可以被删除,以及除 value 和 writable 特性外的其他特性是否可以被修改。
- configurable为默认值或者配置为false时,则该属性相关的配置不能再被更改,也不能被删除
const person = {};
Object.defineProperty(person, 'name', {
value: '小陈',
configurable: false,
})
console.log(person.name);
Object.defineProperty(person, 'name', {
writable: true,
})
Object.defineProperty(person, 'name', {
configurable: true,
})
Object.defineProperty(person, 'name', {
enumerable: true,
})
Object.defineProperty(person, 'name', {
value: '小王'
})
Object.defineProperty(person, 'name', {
get() {
return '小王'
}
})
Object.defineProperty(person, 'name', {
set() {}
})
delete person.name
console.log(person);
- 以上将configurable设为true时,则不会抛出任何错误,最后该属性会被删除。
注意:configurable: true时,可以通过Object.defineProperty给同一属性重复定义配置。同时数据描述符配置属性writable为false,则不能通过赋值方式改变属性的值,但是可以通过Object.defineProperty重复定义的方法改变。
const person = {};
Object.defineProperty(person, 'name', {
value: '小陈',
configurable: true,
writable: false,
})
console.log(person.name);
person.name = '小王';
console.log(person.name);
Object.defineProperty(person, 'name', {
value: '小王',
})
console.log(person.name);
enumerable
- enumerable 定义了对象的属性是否可以在 for…in 循环和 Object.keys() 中被枚举。默认值为 false。
const person = {};
Object.defineProperty(person, "name", {
value: '小陈',
enumerable: true
});
Object.defineProperty(person, "age", {
value: '20',
enumerable: false
});
Object.defineProperty(person, "sex", {
value: '男',
enumerable: true
});
Object.defineProperty(person, "hobby", {
value: '睡觉',
enumerable: true
});
for (var i in person) {
console.log(i);
}
const result = Object.keys(person);
console.log(result);
- 通过propertyIsEnumerable方法确定属性是否可枚举
console.log(person.propertyIsEnumerable('age'));
数据描述符中的属性默认值问题
- 使用点运算符和 Object.defineProperty() 为对象的属性赋值时,描述符中的属性默认值是不同的
var o = {};
o.a = 1;
Object.defineProperty(o, "a", {
value: 1,
writable: true,
configurable: true,
enumerable: true
});
Object.defineProperty(o, "a", { value : 1 });
Object.defineProperty(o, "a", {
value: 1,
writable: false,
configurable: false,
enumerable: false
});
- 即通过点运算符创建的对象属性是可修改可删除可枚举, writable、configurable、enumerable都为true
自定义一个getters和setters
function Archiver() {
var temperature = null;
var archive = [];
Object.defineProperty(this, 'temperature', {
get: function() {
console.log('get!');
return temperature;
},
set: function(value) {
temperature = value;
archive.push({ val: temperature });
}
});
this.getArchive = function() { return archive; };
}
var arc = new Archiver();
arc.temperature;
arc.temperature = 11;
arc.temperature = 13;
arc.getArchive();
Object.defineProperty的应用
监听对象上的多个属性
错误写法
const person = {
name:'小陈',
age:20,
sex:'男',
}
Object.keys(person).forEach(function (key) {
Object.defineProperty(person, key, {
enumerable: true,
configurable: true,
get() {
return person[key]
},
set(val) {
console.log(`对person中的${key}属性进行了修改`)
person[key] = val
}
})
})
console.log(person.age)
这种写法在get/set里写person[key],进入递归调用状态,反复访问属性值。最终造成栈溢出。
正确写法
const person = {
name: '小陈',
age: 20,
sex:'男'
}
function defineProperty(obj, key, val) {
Object.defineProperty(obj, key, {
get() {
console.log(`访问了${key}属性`)
return val
},
set(newVal) {
console.log(`${key}属性被修改为${newVal}了`)
val = newVal
}
})
}
function observer(obj) {
Object.keys(obj).forEach((key) => {
defineProperty(obj, key, obj[key])
})
}
Observer(person);
console.log(person.age);
person.age = 24;
console.log(person.age);
深度监听一个对象
- 借助Object.defineProperty与递归的思想可达到深度监听的效果
function defineProperty(obj, key, val) {
if (typeof val === "object") {
observer(val);
}
Object.defineProperty(obj, key, {
get() {
console.log(`访问了${key}属性`);
return val;
},
set(newVal) {
if (typeof val === 'object') {
observer(key)
}
console.log(`${key}属性被修改为${newVal}了`);
val = newVal;
},
});
}
function observer(obj) {
if (typeof obj !== "object" || obj === null) {
return;
}
Object.keys(obj).forEach((key) => {
defineProperty(obj, key, obj[key]);
});
}
Object.defineProperty的不足
不支持监听数组的变化
let arr = [1, 2, 3]
let obj = {}
Object.defineProperty(obj, 'arr', {
get() {
console.log('触发了get')
return arr
},
set(newVal) {
console.log('触发了set', newVal)
arr = newVal
}
})
console.log(obj.arr)
obj.arr = [1, 2, 3, 4]
obj.arr.push(5)
obj.arr.unshift()
obj.arr.pop()
obj.arr.shift()
当使用Object.defineProperty监听的对象属性是数组时,使用push、unshift、pop、shift、splice, ‘sort’, reverse是监听是触发不了set的。 只要不是重新赋值一个新的数组对象,任何对数组内部的修改都不会触发set方法的执行。
- 在Vue2.x中,通过重写Array原型上的方法解决了这个问题的。
通过重写数组操作方法实现数组监听
function arrMethods() {
const orginalProto = Array.prototype;
const arrayProto = Object.create(orginalProto);
const methodsToPatch = [
'push',
'pop',
'shift',
'unshift',
'splice',
'sort',
'reverse'
]
methodsToPatch.forEach(method => {
arrayProto[method] = function() {
orginalProto[method].apply(this, arguments)
console.log('监听成功', method)
}
})
return arrayProto
}
function observer(data) {
if (Array.isArray(data)) {
data.__proto__ = arrMethods()
for (let i = 0; i < data.length; i++) {
observer(data[i])
}
}
}
const arr = [1, 2, 3, 4]
observer(arr);
arr.push(5);
实现监听对象数组
function arrMethods() {
const orginalProto = Array.prototype;
const arrayProto = Object.create(orginalProto);
const methodsToPatch = [
'push',
'pop',
'shift',
'unshift',
'splice',
'sort',
'reverse'
]
methodsToPatch.forEach(method => {
arrayProto[method] = function() {
orginalProto[method].apply(this, arguments)
console.log('监听成功', method)
}
})
return arrayProto
}
function defineProperty(obj, key, val) {
if (typeof val === "object") {
observer(val);
}
Object.defineProperty(obj, key, {
get() {
console.log(`访问了${key}属性`);
return val;
},
set(newVal) {
if (typeof val === 'object') {
observer(key)
}
console.log(`${key}属性被修改为${newVal}了`);
val = newVal;
},
});
}
function observer(data) {
if (typeof data !== "object" || data === null) {
return;
}
if (Array.isArray(data)) {
data.__proto__ = arrMethods()
for (let i = 0; i < data.length; i++) {
observer(data[i])
}
} else {
Object.keys(data).forEach((key) => {
defineProperty(data, key, data[key]);
});
}
}
const arr = [1, 2, 3, 4]
observer(arr)
arr.push(5)
const person = {
name: '小陈',
age: 20,
sex: '男',
arr: [1, 2, 3, 4]
}
observer(person)
console.log(person.age);
person.age = 24;
console.log(person.age);
person.arr.push(5)
不能监听属性新增和删除操作
const person = {
name: '小陈'
};
let nameValue = person.name
Object.defineProperty(person, 'name', {
configurable: true,
get: function() {
console.log('触发get');
return nameValue
},
set: function(newValue) {
console.log('触发set')
nameValue = newValue
}
})
person.age = 20;
console.log(person);
delete person.age;
console.log(person);
什么样的 a 可以满足 (a === 1 && a === 2 && a === 3) === true 呢?
- 每次访问 a 返回的值都不一样,那么肯定会想到数据劫持。
let current = 0
Object.defineProperty(window, 'a', {
get() {
current++
return current
}
})
console.log(a === 1 && a === 2 && a === 3)
|