Подтвердить что ты не робот

Как вы используете JSON.stringify в пользовательском методе toJSON?

Итак, JSON.stringify предоставляет отличный способ превратить JS-объект, например:

var baz = {"foo":1, "bar":someFunction};

в строку JSON, например:

{"foo":1}

Он делает это с необязательным вторым аргументом, который контролирует, какие поля должны быть сериализованы:

JSON.stringify(baz, ["foo"]);

Это здорово, но есть проблема. Скажем, ваш "баз" на самом деле является свойством другого объекта, и вы хотите сериализовать этот другой объект:

someObject.baz = {"foo":1, "bar":someFunction};
JSON.stringify(someObject, ["baz"]);

Ну, обычно вы просто определяете метод toJSON для baz, например:

someObject.baz = {"foo":1, "bar":someFunction};
someObject.baz.toJSON = function() { /* logic to "toJSON" baz*/ }
JSON.stringify(someObject, ["baz"]);

Теперь, как я упоминал ранее, у нас есть идеальная логика для "toJSON" baz:

someObject.baz.toJSON = function() {
    return JSON.stringify(baz, ["foo"]);
}

но если вы попытаетесь поместить это в свой toJSON, вы получите ошибку рекурсии, потому что stringify вызовет toJSON, который вызовет stringify, который будет...: - (

Вы можете обойти это с помощью взлома:

someObject.baz.toJSON = function() {
    var oldToJON = this.toJSON;
    this.toJSON = null;
    var ret = JSON.stringify(baz, ["foo"]);
    this.toJSON = oldToJON;
    return ret;
}

Но... это просто кажется неправильным. Итак, мой вопрос заключается в следующем: есть ли способ использовать отличную встроенную способность сериализации JSON.stringify внутри метода toJSON объекта (без необходимости скрывать метод toJSON во время операции stringify)?

4b9b3361

Ответ 1

Crockford json2.js говорит:

Метод A toJSON не сериализуется: он возвращает значение, представленное парой имя/значение, которое должно быть сериализовано, или undefined, если ничто не должно быть сериализовано.

Итак, вы просто ожидаете вернуть значение, которое вы хотите сериализовать. В вашем случае baz.toJSON должен просто вернуть часть объекта baz, который вы хотите сериализовать:

someObject.baz.toJSON = function() {
  return { foo: this.foo };
};