我希望通过根据已知的默认设置对其进行分类来降低JSON数据的存储要求。基本上,我想要的是jQuery的.extend()
函数的反转,以便以下测试传递任意JSON兼容的对象:
function test_delta(defaults, delta) {
var current = $.extend(true, {}, defaults, delta);
QUnit.same(get_delta(current, defaults), delta);
}
在我开始编写自己的get_delta()
之前,是否有人知道现有的实施?
答案 0 :(得分:1)
你真正想要的是一个物体差异(erential)算法。
写起来并不难 -
function diff (obj1, obj2) {
var delta = {};
for (var x in obj1) {
if (obj2.hasOwnProperty(x)) {
if (typeof obj2[x] == "object") {
//recurse nested objects/arrays
delta[x] = diff(obj1[x], obj2[x]);
}
else {
//if obj2 doesn't match then - modified attribute
if (obj2[x] != obj1[x]) {
delta[x] = obj1[x];
}
}
}
else {
//obj2 doesn't have this - new attribute
delta[x] = obj1[x];
}
}
return delta;
}
alert(
JSON.stringify(
diff({ hello : 'world', gone : 'fishing' },
{ hello : 'world' })
)
);
//outputs:
{ gone : 'fishing' }
正如您所看到的,这是一个非常基本的实现 - 您可以通过在单独的对象中返回对obj2的添加来扩展它以提供完整的差异。
此代码不是无错误的,对象原型和函数将在不同的浏览器中以不同的方式处理,但它应该足以作为数据结构的演示。
答案 1 :(得分:1)
尝试类似的东西:
jQuery.extend({
deltaExtend: function(deep, target, defaults, delta){
var result = jQuery.extend.apply(jQuery, arguments);
jQuery(result).data('delta', delta);
jQuery(result).data('defaults', defaults);
return result;
}
});
用法:
var result = $.deltaExtend(true, {}, defaults, delta);
$(result).data('delta') //returns delta object
$(result).data('defaults') //returns default object
它也可以调整以使其适用于N个对象,只需要更多的思考。
答案 2 :(得分:1)
我知道与主题启动器相关的时间有点晚,但您可能需要查看Underscore.js中的_.omit(object, *keys)
函数。就是这样。