Javascript toJSON自定义方法的Stackoverflow错误

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.因此,如果第一个调用第二个,则生成循环.

问题

  1. (一般)为什么这个设计选择?toJSON是一种保留的特殊关键字?
  2. (具体)我解决了将.toJSON名称更改为的stackoverflow错误.display.不那么优雅.还有其他解决方案吗?

and*_*lrc 6

认为这是因为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)


lrs*_*jng 3

问题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)