Alb*_*aro 7 javascript oop json design-patterns
脚本
读完这个答案后,我意识到我可以从JSON文字开始创建对象.
所以我猜我只能使用这个有用的JSON方法做相反的事情:
JSON.stringify(myObject)
.
所以我做了如下:
function MyObject(id, value, desc)
{
this.id = id;
this.value = value;
this.desc = desc;
this.toJSON = function()
{
return JSON.stringify(this);
}
}
Run Code Online (Sandbox Code Playgroud)
但是当我运行这个东西(演示)Maximum call stack size exceeded
时会发生错误.
谷歌搜索后,我发现两个引用解释了这种行为:
如果我做对了,就会.toJSON
超越.stringify
.因此,如果第一个调用第二个,则生成循环.
问题
toJSON
是一种保留的特殊关键字?.toJSON
名称更改为的stackoverflow错误.display
.不那么优雅.还有其他解决方案吗?认为这是因为toJSON
半保留:stringify
将检查对象并查看它是否有一个被调用的方法toJSON
,然后尝试调用它来对结果进行字符串.
解决方法可以是:(不确定此代码的可靠性)
var obj = {
value: 1,
name: "John",
toJSON: function() {
var ret,
fn = this.toJSON;
delete this.toJSON;
ret = JSON.stringify(this);
this.toJSON = fn;
return ret;
}
}
Run Code Online (Sandbox Code Playgroud)
用法:
obj.toJSON(); // "{\"value\":1,\"name\":\"John\"}"
obj.lastName = "Smith";
obj.toJSON(); // "{\"value\":1,\"name\":\"John\",\"lastName\":\"Smith\"}"
Run Code Online (Sandbox Code Playgroud)
也许使用clousure有点漂亮:(然后我想我可以说它是安全的)
var obj = {
value: 1,
name: "John",
toJSON: (function() {
function fn() {
var ret;
delete this.toJSON;
ret = JSON.stringify(this);
this.toJSON = fn;
return ret;
}
return fn;
})()
}
Run Code Online (Sandbox Code Playgroud)
所以在阅读@ filmor的评论之后,我想到了另一种处理这个问题的方法.不是很漂亮,但它的工作原理.
使用Function.caller我可以检测是否fn
使用了JSON.stringify
var obj = {
value: 1,
name: "John",
toJSON: (function() {
return function fn() {
var ret;
delete this.toJSON;
ret = JSON.stringify(this);
if ( fn.caller === JSON.stringify ) {
ret = JSON.parse( ret );
}
this.toJSON = fn;
return ret;
}
})()
}
Run Code Online (Sandbox Code Playgroud)
问题1、是否toJSON
保留?
我不确定它是否保留,但例如本机 Date 对象使用 toJSON 创建字符串化日期表示:
(new Date()).toJSON(); // -> "2012-10-20T01:58:21.427Z"
JSON.stringify({d: new Date()}); // -> {"d":"2012-10-20T01:58:21.427Z"}"
Run Code Online (Sandbox Code Playgroud)
问题2,一个简单的解决方案:
创建忽略 toJSON 方法的自定义 stringify 函数(您可以将其添加到现有的 global 中JSON
):
JSON.customStringify = function (obj) {
var fn = obj.toJSON;
obj.toJSON = undefined;
var json = JSON.stringify(obj);
obj.toJSON = fn;
return json;
}
Run Code Online (Sandbox Code Playgroud)
现在它很容易在您的所有对象中使用:
function MyObject(id, value, desc)
{
this.id = id;
this.value = value;
this.desc = desc;
this.toJSON = function()
{
return JSON.customStringify(this);
}
}
Run Code Online (Sandbox Code Playgroud)
为了使它更容易另外添加:
JSON.customStringifyMethod = function () {
return JSON.customStringify(this);
}
Run Code Online (Sandbox Code Playgroud)
现在你的对象可能看起来像:
function MyObject(id, value, desc)
{
this.id = id;
this.value = value;
this.desc = desc;
this.toJSON = JSON.customStringifyMethod;
}
Run Code Online (Sandbox Code Playgroud)