JavaScript中,我们希望别人无法修改我们创建的对象。比如,代码库的作者很可能想锁定核心库的某些部分来保证它们不被意外地修改。ES5中引入了三种锁定修改的级别:防止扩展preventExtensions
、密封seal
、冻结frezze
。
正常对象:
可以添加属性和方法,所有属性和方法可以被删除、修改。
var person = { name: "Nicholas"};
防止扩展:
禁止为对象添加属性和方法。但已存在的属性和方法可以被修改或删除。
实施操作:Object.preventExtensions()
检测是否应用该操作:Object.isExtensible()
Object.preventExtensions(person);console.log(Object.isExtensible(person)); // falseperson.age = 25; // 防止扩展console.log(person.age); // undefined
密封:
禁止为对象删除已存在的属性和方法。被密封的对象也是不可扩展的。
实施操作:Object.seal()
检测是否应用该操作:Object.isSealed()
Object.seal(person);console.log(Object.isExtensible(person)); // falseconsole.log(Object.isSealed(person)); // trueperson.age = 25; // 防止扩展console.log(person.age); // undefineddelete person.name; // 防止删除console.log(person.name); // Nicholas
冻结:
禁止为对象修改已存在的属性和方法。所有字段均只读。被冻结的对象也是不可扩展和密封的。
实施操作:Object.freeze()
检测是否应用该操作:Object.isFrozen()
Object.freeze(person);console.log(Object.isExtensible(person)); // falseconsole.log(Object.isSealed(person)); // trueconsole.log(Object.isFrozen(person)); // trueperson.age = 25; // 防止扩展console.log(person.age); // undefineddelete person.name; // 防止删除console.log(person.name); // Nicholasperson.name = "Greg" // 防止修改console.log(person.name); // Nicholas
注意:
- 对于同一个对象,
防止扩展-->密封-->冻结
这种操作是不可逆的。一旦该对象被冻结,是无法恢复到防止扩展或密封状态的。一旦该对象被密封,是无法恢复到防止扩展状态的。一旦对象被锁定,它将无法解锁。
Object.freeze(person);person.name = "Greg"; // 防止修改console.log(person.name); // NicholasObject.preventExtensions(person); // 无法回退person.name = "Greg"; // 防止修改console.log(person.name); // Nicholas
Object.seal()
方法,会在一个现有对象上调用Object.preventExtensions()
,并把所有属性标记为configurable: false
。Object.freeze()
方法,会在一个现有对象上调用Object.seal()
,并把所有属性标记为writable: false
。- 深度冻结一个对象是指,首先在这个对象上调用
Object.freeze()
,然后遍历它引用的所有对象,并且在这些对象上调用Object.freeze()
。
总结:
添加 | 删除 | 修改 | 实施操作 | 检测是否应用 | |
---|---|---|---|---|---|
防止扩展 | x | v | v | Object.preventExtensions() | Object.isExtensible() |
密封 | x | x | v | Object.seal() | Object.isSealed() |
冻结 | x | x | x | Object.freeze() | Object.isFrozen() |
参考自《编写可维护的JavaScript》、《你不知道的JavaScript(上卷)》