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

Javascript 在JSON.parse()之后保留属性属性(可写、可配置),javascript,json,Javascript,Json,假设我正在其他地方创建一个对象,并以某种方式将其传递给我的模块。也许它是在服务器上的node.js中创建的,也许它是在不同的模块中创建的,不管出于什么原因,我正在JSON.stringify()对它进行加密并传递序列化版本(特别是如果它来自服务器)。但我希望这个特殊属性是不可变的: var foo = {}; Object.defineProperty(foo, 'bar', { value: 'bar', writeable: false, enumerable: t

假设我正在其他地方创建一个对象,并以某种方式将其传递给我的模块。也许它是在服务器上的
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
、嵌套对象等的访问器类型描述符

在键上使用标志 另一个想法是通过输入和输出过程中的转换,将可写性编码到keyname中:

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

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

JSON与语言无关。这个元数据是特定于Javascript的。这非常有意义。除了单独发送元数据并在解析后重建元数据之外,还有没有其他序列化技术可以达到预期的效果?据我所知,没有标准,您必须设计一个自定义解决方案。谢谢,我试图避免它,但我将只编写一个自定义的
toString
和解析器来处理它。不用麻烦了。。当foo是字符串格式时,任何字符串操作都可以覆盖/替换您将进行的每个配置。。如果您想“保护”fii,只需在JSON.parse之后调用Object.defineProperty即可