在 JSON.parse() 之后保留属性属性(可写、可配置)

Preserve property attributes (writeable, configurable) after JSON.parse()

本文关键字:属性 可写 配置 之后 JSON parse 保留      更新时间:2023-09-26

假设我正在其他地方创建一个对象并将其以某种方式传递给我的模块。也许它是在 node.js 年在服务器上创建的,也许它是在不同的模块中创建的,无论出于何种原因,我JSON.stringify()它并通过序列化版本(特别是如果它来自服务器)。但我希望这个特定的属性是不可变的:

var foo = {};
Object.defineProperty(foo, 'bar', {
    value: 'bar',
    writeable: false,
    enumerable: true
});
console.log(foo.bar); //bar
foo.bar = 'foo';      //fails, throws err in strict
console.log(foo.bar); //still bar
var fii = JSON.parse(JSON.stringify(foo));
console.log(fii.bar); //still bar
fii.bar = 'foo';      //succeeds      
console.log(fii.bar); //now foo

有没有办法保留此元数据,以便bar属性不可变而无需单独发送?

您可以字符串化属性描述符,如

// set up object to be serialized--from OP question
var foo = {};
Object.defineProperty(foo, 'bar', {
    value: 'bar',
    writable: false,
    enumerable: true
});
// create an object of property descriptors
var descriptors = {};
Object.keys(foo).forEach(function(key) {
    descriptors[key] = Object.getOwnPropertyDescriptor(foo, key);
});
var json = JSON.stringify(descriptors);
// "{"bar":{"value":"bar","writable":false,"enumerable":true,"configurable":false}}"

现在,重新构建:

descriptors = JSON.parse(json);
foo = {};
Object.defineProperties(foo, descriptors);

当然,这不适用于包含get和/或set、嵌套对象等的访问器类型描述符。

在键上使用标志

另一个想法是通过进出途中的转换将可写性编码到键名中:

// Mark unwritable properties by changing keyname to 'prop:unwritable'.
function markImmutable(obj) {
    for (var p in obj) {
        if (!Object.getOwnPropertyDescriptor(obj, p).writable) {
            obj[p + ':unwritable'] = obj[p];
            delete obj[p];
        }
    }
}
markImmutable(foo)
> Object {bar:unwritable: "bar"}

然后将其字符串化,并在解析后进行反向转换。