我的应用程序有大量对象,我将它们字符串化并将它们保存到磁盘。不幸的是,当数组中的对象被操作,有时被替换时,对象的属性以不同的顺序列出(它们的创建顺序?)。当我对数组执行 JSON.stringify() 并保存它时,diff 会显示以不同顺序列出的属性,这在尝试使用 diff 和合并工具进一步合并数据时很烦人。
理想情况下,我想在执行字符串化之前按字母顺序对对象的属性进行排序,或者作为字符串化操作的一部分。在许多地方都有用于操作数组对象的代码,并且更改这些代码以始终以明确的顺序创建属性是很困难的。
欢迎提出建议!
一个精简的例子:
obj = {}; obj.name="X"; obj.os="linux";
JSON.stringify(obj);
obj = {}; obj.os="linux"; obj.name="X";
JSON.stringify(obj);
这两个 stringify 调用的输出不同,并显示在我的数据的差异中,但我的应用程序并不关心属性的顺序。对象的构造方式和位置很多。
最佳答案
更简单、现代且当前浏览器支持的方法就是这样:
JSON.stringify(sortMyObj, Object.keys(sortMyObj).sort());
但是,此方法确实会删除任何未引用的嵌套对象,并且不适用于数组中的对象。如果你想要这样的输出,你也需要展平排序对象:
{"a":{"h":4,"z":3},"b":2,"c":1}
你可以这样做:
var flattenObject = function(ob) {
var toReturn = {};
for (var i in ob) {
if (!ob.hasOwnProperty(i)) continue;
if ((typeof ob[i]) == 'object') {
var flatObject = flattenObject(ob[i]);
for (var x in flatObject) {
if (!flatObject.hasOwnProperty(x)) continue;
toReturn[i + '.' + x] = flatObject[x];
}
} else {
toReturn[i] = ob[i];
}
}
return toReturn;
};
var myFlattenedObj = flattenObject(sortMyObj);
JSON.stringify(myFlattenedObj, Object.keys(myFlattenedObj).sort());
要以您可以自己调整的方式以编程方式执行此操作,您需要将对象属性名称推送到一个数组中,然后按字母顺序对数组进行排序并遍历该数组(将按正确的顺序)并从中选择每个值按该顺序排列的对象。 “hasOwnProperty”也被选中,所以你肯定只有对象自己的属性。这是一个例子:
var obj = {"a":1,"b":2,"c":3};
function iterateObjectAlphabetically(obj, callback) {
var arr = [],
i;
for (i in obj) {
if (obj.hasOwnProperty(i)) {
arr.push(i);
}
}
arr.sort();
for (i = 0; i < arr.length; i++) {
var key = obj[arr[i]];
//console.log( obj[arr[i]] ); //here is the sorted value
//do what you want with the object property
if (callback) {
// callback returns arguments for value, key and original object
callback(obj[arr[i]], arr[i], obj);
}
}
}
iterateObjectAlphabetically(obj, function(val, key, obj) {
//do something here
});
同样,这应该保证您按字母顺序遍历。
最后,用最简单的方法,这个库将递归地允许你对传入的任何 JSON 进行排序:https://www.npmjs.com/package/json-stable-stringify
var stringify = require('json-stable-stringify');
var obj = { c: 8, b: [{z:6,y:5,x:4},7], a: 3 };
console.log(stringify(obj));
输出
{"a":3,"b":[{"x":4,"y":5,"z":6},7],"c":8}
https://stackoverflow.com/questions/16167581/