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

时间:2014-11-27 01:46:12

标签: javascript json

让我们说我在其他地方创建了一个对象并以某种方式将它传递给我的模块。也许它是在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属性不可变而不单独发送?

1 个答案:

答案 0 :(得分:0)

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

// 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"}

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