我有两个对象:
{A: 10, B: 20, C: 30}
{A: 10, B: 22, C: 30}
如您所见:除了一件事之外,几乎相等:B
的键值不同。
我如何才能了解someNewArr
键值差异?
像someNewArr
:{B: 22}
(我从第二个对象获取值)
我正在使用 Angular ,我的意思是这样的:
var compareTwoObjects = function(initialObj, editedObj) {
var resultArr = [];
angular.forEach(initialObj, function(firstObjEl, firstObjInd) {
angular.forEach(editedObj, function(secondObjEl, secondObjInd) {
if (firstObjEl.key === secondObjEl.key && firstObjEl.value !== secondObjEl.value){
resultArr.push({firstObjEl.key: secondObjEl.value});
}
})
});
});
最佳答案
递归差异
大约3年后,我很高兴为这个问题提供一个新的答案。
我们从两个不同的对象开始
const x =
{ a: 1, b: 2, c: 3 }
const y =
{ a: 1, b: 3, d: 4 }
console.log (diff (x, y))
// => ???
这两个对象具有相同的a
属性。 b
属性不相同。仅x
具有c
属性,仅y
具有d
属性。那么???
应该是什么呢?
从diff
的 Angular 来看,我们输入对象a
和b
之间的关系可以是完全任意的。为了传达哪个对象做出了贡献,diff
分配了描述符left
和right
console.log (diff (x, y))
// { b: { left: 2, right: 3 }, c: { left: 3 }, d: { right: 4 } }
在上面的输出中,我们可以看到
- 哪些属性不同–
b
,c
和d
* 哪个对象造成了差异-left
和/或right
* 是“不同”的值-例如,左边的b
的值为2,右边的b
的值为3;或左侧c
的值为3,右侧c
的值为undefined
在开始执行此功能之前,我们将首先研究一个更复杂的场景,该场景涉及深度嵌套的对象
const x =
{ a: { b: { c: 1, d: 2, e: 3 } } }
const y =
{ a: { b: { c: 1, d: 3, f: 4 } } }
console.log (diff (x, y))
// { a: { b: { d: { left: 2, right: 3 }, e: { left: 3 }, f: { right: 4 } } } }
正如我们在上面看到的,diff
返回与我们的输入匹配的结构。最后,我们期望两个相同的对象的diff
返回“空”结果
const x1 =
{ a: 1, b: { c: { d: 2 } } }
const x2 =
{ a: 1, b: { c: { d: 2 } } }
console.log (diff (x1, x2))
// {}
上面我们描述了一个diff
函数,该函数不关心输入给定的输入对象。 “左”对象可以包含键,而“右”对象不包含键,反之亦然,但是我们仍然必须从任一侧检测更改。从高层次开始,这就是我们解决问题的方式
const diff = (x = {}, y = {}) =>
merge
( diff1 (x, y, "left")
, diff1 (y, x, "right")
)
差异1
我们使用被称为“左”关系的diff1
进行“单侧”比较,然后将输入对象反转为“右”关系的另一侧进行差异,然后我们将两个结果一起merge
对于我们来说,我们的工作分为易于完成的任务。 diff1
仅需要检测一半的必要更改,而merge
只是将结果组合在一起。我们将从diff1
开始
const empty =
{}
const isObject = x =>
Object (x) === x
const diff1 = (left = {}, right = {}, rel = "left") =>
Object.entries (left)
.map
( ([ k, v ]) =>
isObject (v) && isObject (right[k])
? [ k, diff1 (v, right[k], rel) ]
: right[k] !== v
? [ k, { [rel]: v } ]
: [ k, empty ]
)
.reduce( (acc, [ k, v ]) =>
v === empty ? acc : { ...acc, [k]: v }
, empty)
diff1
接受两个输入对象和一个关系描述符rel
。该描述符默认为"left"
,它是比较的默认“方向”。在下面,请注意diff1
仅提供我们所需结果的一半。在第二次调用diff1
时反转参数将提供另一半。
const x =
{ a: 1, b: 2, c: 3 }
const y =
{ a: 1, b: 3, d: 4 }
console.log (diff1 (x, y, "left"))
// { b: { left: 2 }, c: { left: 3 } }
console.log (diff1 (y, x, "right"))
// { b: { right: 3 }, d: { right: 4 } }
同样值得注意的是,关系标签"left"
和"right"
是用户可定义的。例如,如果您要比较的对象之间具有已知的关系,并且希望在diff输出中提供更多描述性标签...
const customDiff = (x = {}, y = {}) =>
merge
( diff1 (x, y, "original")
, diff1 (y, x, "modified")
)
customDiff( { host: "localhost", port: 80 } , { host: "127.0.0.1", port: 80 } )
// { host: { original: 'localhost', modified: '127.0.0.1' } }
在上面的示例中,使用程序其他区域的输出可能会更容易,因为标签original
和modified
比left
和right
更具描述性。
合并
剩下的全部就是将两个半差异合并成一个完整的结果。我们的merge
函数也可以正常工作,并接受任何两个对象作为输入。
const x = { a: 1, b: 1, c: 1 }
const y = { b: 2, d: 2 }
console.log (merge (x, y))
// { a: 1, b: 2, c: 1, d: 2 }
如果每个对象都包含一个其值也是一个对象的属性,则merge
也会重复出现并合并嵌套的对象。
const x = { a: { b: { c: 1, d: 1 } } }
const y = { a: { b: { c: 2, e: 2 } }, f: 2 }
console.log (merge (x, y))
// { a: { b: { c: 2, d: 1, e: 2 } }, f: 2 }
下面我们将意图编码为merge
const merge = (left = {}, right = {}) =>
Object.entries (right)
.reduce( (acc, [ k, v ]) =>
isObject (v) && isObject (left [k])
? { ...acc, [k]: merge (left [k], v) }
: { ...acc, [k]: v }
, left
)
这就是整个套件和caboodle!展开下面的代码片段,以在您自己的浏览器中运行代码演示
const empty =
{}
const isObject = x =>
Object (x) === x
const diff1 = (left = {}, right = {}, rel = "left") =>
Object.entries (left)
.map
( ([ k, v ]) =>
isObject (v) && isObject (right[k])
? [ k, diff1 (v, right[k], rel) ]
: right[k] !== v
? [ k, { [rel]: v } ]
: [ k, empty ]
)
.reduce
( (acc, [ k, v ]) =>
v === empty
? acc
: { ...acc, [k]: v }
, empty
)
const merge = (left = {}, right = {}) =>
Object.entries (right)
.reduce
( (acc, [ k, v ]) =>
isObject (v) && isObject (left [k])
? { ...acc, [k]: merge (left [k], v) }
: { ...acc, [k]: v }
, left
)
const diff = (x = {}, y = {}) =>
merge
( diff1 (x, y, "left")
, diff1 (y, x, "right")
)
const x =
{ a: { b: { c: 1, d: 2, e: 3 } } }
const y =
{ a: { b: { c: 1, d: 3, f: 4 } } }
console.log (diff (x, y))
// { a: { b: { d: { left: 2, right: 3 }, e: { left: 3 }, f: { right: 4 } } } }
console.log (diff (diff (x,y), diff (x,y)))
// {}
备注
当我们回顾diff
函数时,我想强调其设计的一个重要部分。大部分工作由merge
函数处理,该函数与diff
完全分开,但tough nut to crack本身是独立的。由于我们将关注点分为单个功能,因此现在可以很容易地在程序的其他区域中重用它们。我们想要diff
的地方,就得到了它,并且免费获得了直观的深度merge
功能。
额外:支持数组
我们的diff
函数非常方便,因为它可以爬行深度嵌套的对象,但是如果我们的对象属性之一是数组怎么办?如果我们可以使用相同的技术来比较数组,那将是很好的。
支持此功能需要对上面的代码进行不重要的更改。但是,大多数结构和推理都保持不变。例如,diff
完全不变
// unchanged
const diff = (x = {}, y = {}) =>
merge
( diff1 (x, y, "left")
, diff1 (y, x, "right")
)
为了支持merge
中的数组,我们引入了一个突变帮助程序mut
,它将一个[ key, value ]
对分配给一个给定的对象o
。数组也被视为对象,因此我们可以使用相同的mut
函数更新数组和对象
const mut = (o, [ k, v ]) =>
(o [k] = v, o)
const merge = (left = {}, right = {}) =>
Object.entries (right)
.map
( ([ k, v ]) =>
isObject (v) && isObject (left [k])
? [ k, merge (left [k], v) ]
: [ k, v ]
)
.reduce (mut, left)
浅合并工作按预期方式
const x =
[ 1, 2, 3, 4, 5 ]
const y =
[ , , , , , 6 ]
const z =
[ 0, 0, 0 ]
console.log (merge (x, y))
// [ 1, 2, 3, 4, 5, 6 ]
console.log (merge (y, z))
// [ 0, 0, 0, <2 empty items>, 6 ]
console.log (merge (x, z))
// [ 0, 0, 0, 4, 5, 6 ]
以及深层合并
const x =
{ a: [ { b: 1 }, { c: 1 } ] }
const y =
{ a: [ { d: 2 }, { c: 2 }, { e: 2 } ] }
console.log (merge (x, y))
// { a: [ { b: 1, d: 2 }, { c: 2 }, { e: 2 } ] }
在diff1
中支持数组要困难得多
const diff1 = (left = {}, right = {}, rel = "left") =>
Object.entries (left)
.map
( ([ k, v ]) =>
isObject (v) && isObject (right[k])
? [ k, diff1 (v, right[k], rel) ]
: right[k] !== v
? [ k, { [rel]: v } ]
: [ k, {} ]
)
.filter
( ([ k, v ]) =>
Object.keys (v) .length !== 0
)
.reduce
( mut
, isArray (left) && isArray (right) ? [] : {}
)
但是有了这些更改之后,我们现在就可以深度比较包含数组的对象,甚至可以比较包含对象的数组!
const x =
{ a: 1, b: [ { c: 1 }, { d: 1 }, { e: 1 } ] }
const y =
{ a: 1, b: [ { c: 2 }, { d: 1 }, 5, 6 ], z: 2 }
console.log (diff (x, y))
// { b:
// [ { c: { left: 1, right: 2 } }
// , <1 empty item>
// , { left: { e: 1 }, right: 5 }
// , { right: 6 }
// ]
// , z: { right: 2 }
// }
由于diff1
会根据其输入类型仔细更改其行为,因此我们免费获得数组差异
const x =
[ 1, 2, 3, 4 ]
const y =
[ 1, 2, 9 ]
const z =
[ 1, 2, 9 ]
console.log (diff (x, y))
// [ <2 empty items>, { left: 3, right: 9 }, { left: 4 } ]
console.log (diff (y, z))
// []
在下面的浏览器中运行完整程序
const isObject = x =>
Object (x) === x
const isArray =
Array.isArray
const mut = (o, [ k, v ]) =>
(o [k] = v, o)
const diff1 = (left = {}, right = {}, rel = "left") =>
Object.entries (left)
.map
( ([ k, v ]) =>
isObject (v) && isObject (right[k])
? [ k, diff1 (v, right[k], rel) ]
: right[k] !== v
? [ k, { [rel]: v } ]
: [ k, {} ]
)
.filter
( ([ k, v ]) =>
Object.keys (v) .length !== 0
)
.reduce
( mut
, isArray (left) && isArray (right) ? [] : {}
)
const merge = (left = {}, right = {}) =>
Object.entries (right)
.map
( ([ k, v ]) =>
isObject (v) && isObject (left [k])
? [ k, merge (left [k], v) ]
: [ k, v ]
)
.reduce (mut, left)
const diff = (x = {}, y = {}) =>
merge
( diff1 (x, y, "left")
, diff1 (y, x, "right")
)
const x =
{ a: 1, b: [ { c: 1 }, { d: 1 }, { e: 1 } ] }
const y =
{ a: 1, b: [ { c: 2 }, { d: 1 }, 5, 6 ], z: 2 }
console.log (diff (x, y))
// { b:
// [ { c: { left: 1, right: 2 } }
// , <1 empty item>
// , { left: { e: 1 }, right: 5 }
// , { right: 6 }
// ]
// , z: { right: 2 }
// }
浅差异
此答案的previous version提供了一个对象diff
函数,用于比较具有相同键的对象并比较具有不同键的对象,但是没有一种解决方案对嵌套对象递归执行diff。
递归交点
在this related Q&A中,我们采用两个输入对象并计算递归intersect
而不是diff
。