これは本当にきちんとしたシンプルなもので(少なくとも私はそう信じています:))、日付を操作したり、toJSONなどのブラウザーのネイティブ関数をオーバーロードしたりする必要はありません(参照:JSONをJavaScriptで文字列化し、タイムゾーンを保持する方法、丁寧なShawson)
JSON.stringifyに置換機能を渡して、内容を文字列化します!!! このようにして、時間や分の差分やその他の操作を行う必要がありません。
中間結果を確認するためにconsole.logsを入れたので、何が行われていて、再帰がどのように機能しているかが明確です。これは、注目に値する何かを明らかにします:replacerへの値パラメーターは、ISO日付形式に既に変換されています:)。これを使用して、元のデータを操作します。
var replacer = function(key, value)
{
var returnVal = value;
if(this[key] instanceof Date)
{
console.log("replacer called with key - ", key, " value - ", value, this[key]);
returnVal = this[key].toString();
/* Above line does not strictly speaking clone the date as in the cloned object
* it is a string in same format as the original but not a Date object. I tried
* multiple things but was unable to cause a Date object being created in the
* clone.
* Please Heeeeelp someone here!
returnVal = new Date(JSON.parse(JSON.stringify(this[key]))); //OR
returnVal = new Date(this[key]); //OR
returnVal = this[key]; //careful, returning original obj so may have potential side effect
*/
}
console.log("returning value: ", returnVal);
/* if undefined is returned, the key is not at all added to the new object(i.e. clone),
* so return null. null !== undefined but both are falsy and can be used as such*/
return this[key] === undefined ? null : returnVal;
};
ab = {prop1: "p1", prop2: [1, "str2", {p1: "p1inner", p2: undefined, p3: null, p4date: new Date()}]};
var abstr = JSON.stringify(ab, replacer);
var abcloned = JSON.parse(abstr);
console.log("ab is: ", ab);
console.log("abcloned is: ", abcloned);
/* abcloned is:
* {
"prop1": "p1",
"prop2": [
1,
"str2",
{
"p1": "p1inner",
"p2": null,
"p3": null,
"p4date": "Tue Jun 11 2019 18:47:50 GMT+0530 (India Standard Time)"
}
]
}
Note p4date is string not Date object but format and timezone are completely preserved.
*/
2009-09-28T10:00:00Z
時間内に同じ瞬間を表すものではないとしてMon Sep 28 10:00:00 UTC+0200 2009
。Z
でISO 8601の日付はUTCを意味し、UTCで10時はある時間の異なる瞬間 0200で10時まで。日付を正しいタイムゾーンでシリアル化することは1つのことですが、明確に、客観的に間違っている表現に日付をシリアル化するのを支援するように求めています。