一、Object.seal()
Object.seal()方法用于密封一个对象,即将对象设置为不可扩展,同时将对象的所有自有属性都设置为不可配置(包括Symbol值的属性)。也就是说,不能给对象添加新的属性和方法,也不能删除现有的属性和方法、不能修改现有属性和方法的配置。但如果对象的属性和方法是可写的,那该属性和方法仍然可以修改。
语法:
Object.seal(obj);
参数:
obj要被密封的对象。
返回值:
被密封的对象。
该操作不会影响从原型对象继承来的属性和方法,即只影响自有的属性和方法。一旦对象被密封,则不能修改其属性和方法的配置,一个数据属性不能被重新定义成访问器属性,也不能从访问器属性修改为数据属性。
在ES5中,如果传递给方法的参数不是一个对象,会抛出TypeError异常。
在ES6中,如果传递给方法的参数不是一个对象,则会被视为已被密封的普通对象,直接返回它。
示例一:密封一个对象
let obj = { a: 1, b: function () { console.log(2); }, [Symbol('c')]: 3 };
Object.getOwnPropertyDescriptors(obj);
/*
{ a: {value: 1, writable: true, enumerable: true, configurable: true}, b: {value: ƒ, writable: true, enumerable: true, configurable: true}, Symbol(c): {value: 3, writable: true, enumerable: true, configurable: true} } */
Object.seal(obj);
Object.getOwnPropertyDescriptors(obj);
/*
{
a: {value: 1, writable: true, enumerable: true, configurable: false},
b: {value: ƒ, writable: true, enumerable: true, configurable: false},
Symbol(c): {value: 3, writable: true, enumerable: true, configurable: false}
}
*/
二、Object.freeze()
Object.freeze()
方法可以冻结一个对象。一个被冻结的对象再也不能被修改;冻结了一个对象则不能向这个对象添加新的属性,不能删除已有属性,不能修改该对象已有属性的可枚举性、可配置性、可写性,以及不能修改已有属性的值。此外,冻结一个对象后该对象的原型也不能被修改。freeze()
返回和传入的参数相同的对象。
语法:
Object.freeze(obj)
参数:
obj
要被冻结的对象。
返回值:
被冻结的对象
var obj = { prop: function() {}, foo: 'bar' }; // 新的属性会被添加, 已存在的属性可能会被修改或移除 obj.foo = 'baz'; obj.lumpy = 'woof'; delete obj.prop;
// 作为参数传递的对象与返回的对象都被冻结,所以不必保存返回的对象(因为两个对象全等) var o = Object.freeze(obj); o === obj; // true Object.isFrozen(obj); // === true // 现在任何改变都会失效 obj.foo = 'quux'; // 静默地不做任何事 // 静默地不添加此属性 obj.quaxxor = 'the friendly duck'; // 在严格模式,如此行为将抛出 TypeErrors function fail(){ 'use strict'; obj.foo = 'sparky'; // throws a TypeError delete obj.quaxxor; // 返回true,因为quaxxor属性从来未被添加 obj.sparky = 'arf'; // throws a TypeError } fail(); // 试图通过 Object.defineProperty 更改属性 // 下面两个语句都会抛出 TypeError. Object.defineProperty(obj, 'ohai', { value: 17 }); Object.defineProperty(obj, 'foo', { value: 'eit' }); // 也不能更改原型 // 下面两个语句都会抛出 TypeError. Object.setPrototypeOf(obj, { x: 20 }) obj.__proto__ = { x: 20 }
冻结数组
let a = [0]; Object.freeze(a); // 现在数组不能被修改了. a[0]=1; // fails silently a.push(2); // fails silently // In strict mode such attempts will throw TypeErrors function fail() { "use strict" a[0] = 1; a.push(2); } fail();
被冻结的对象是不可变的。但也不总是这样。下例展示了冻结对象不是常量对象(浅冻结)。
obj1 = { internal: {} }; Object.freeze(obj1); obj1.internal.a = 'aValue'; obj1.internal.a; // 'aValue'
要使对象不可变,需要递归冻结每个类型为对象的属性(深冻结)。
// 深冻结函数. function deepFreeze(obj) { // 取回定义在obj上的属性名 var propNames = Object.getOwnPropertyNames(obj); // 在冻结自身之前冻结属性 propNames.forEach(function(name) { var prop = obj[name]; // 如果prop是个对象,冻结它 if (typeof prop == 'object' && prop !== null) deepFreeze(prop); }); // 冻结自身(no-op if already frozen) return Object.freeze(obj); } obj2 = { internal: {} }; deepFreeze(obj2); obj2.internal.a = 'anotherValue'; obj2.internal.a; // undefined
在ES5中,如果这个方法的参数不是一个对象(一个原始值),那么它会导致 TypeError
。在ES2015中,非对象参数将被视为要被冻结的普通对象,并被简单地返回。
> Object.freeze(1) TypeError: 1 is not an object // ES5 code > Object.freeze(1) 1 // ES2015 code
用Object.seal()
密封的对象可以改变它们现有的属性。使用Object.freeze()
冻结的对象中现有属性是不可变的。