我采用了一种更通用的方法,尽管在想法上与@Cerbrus和@Kasper Moerch的方法相似。我创建了一个函数,它接受一个谓词来确定两个对象是否相等(这里我们忽略了该 $$hashKey
属性,但它可以是任何东西)并返回一个函数,该函数根据该谓词计算两个列表的对称差异:
a = [{ value:"4a55eff3-1e0d-4a81-9105-3ddd7521d642", display:"Jamsheer"}, { value:"644838b3-604d-4899-8b78-09e4799f586f", display:"Muhammed"}, { value:"b6ee537a-375c-45bd-b9d4-4dd84a75041d", display:"Ravi"}, { value:"e97339e1-939d-47ab-974c-1b68c9cfb536", display:"Ajmal"}, { value:"a63a6f77-c637-454e-abf2-dfb9b543af6c", display:"Ryan"}]
b = [{ value:"4a55eff3-1e0d-4a81-9105-3ddd7521d642", display:"Jamsheer", $$hashKey:"008"}, { value:"644838b3-604d-4899-8b78-09e4799f586f", display:"Muhammed", $$hashKey:"009"}, { value:"b6ee537a-375c-45bd-b9d4-4dd84a75041d", display:"Ravi", $$hashKey:"00A"}, { value:"e97339e1-939d-47ab-974c-1b68c9cfb536", display:"Ajmal", $$hashKey:"00B"}]
var makeSymmDiffFunc = (function() {
var contains = function(pred, a, list) {
var idx = -1, len = list.length;
while (++idx < len) {if (pred(a, list[idx])) {return true;}}
return false;
};
var complement = function(pred, a, b) {
return a.filter(function(elem) {return !contains(pred, elem, b);});
};
return function(pred) {
return function(a, b) {
return complement(pred, a, b).concat(complement(pred, b, a));
};
};
}());
var myDiff = makeSymmDiffFunc(function(x, y) {
return x.value === y.value && x.display === y.display;
});
var result = myDiff(a, b); //=> {value="a63a6f77-c637-454e-abf2-dfb9b543af6c", display="Ryan"}
与 Cerebrus 的方法相比,它有一个小优势(就像 Kasper Moerch 的方法一样),它可以提前逃脱;如果找到匹配项,则不会检查列表的其余部分。如果我curry
手头有一个函数,我会做一些不同的事情,但这工作正常。
解释
一个评论要求为初学者提供更详细的解释。这是一个尝试。
我们将以下函数传递给makeSymmDiffFunc
:
function(x, y) {
return x.value === y.value && x.display === y.display;
}
这个函数是我们决定两个对象相等的方式。像所有返回true
or 的函数一样false
,它可以被称为“谓词函数”,但这只是术语。重点是它makeSymmDiffFunc
配置了一个函数,该函数接受两个对象,true
如果我们认为它们相等false
则返回,否则返回。
使用它,makeSymmDiffFunc
(阅读“制作对称差分函数”)返回一个新函数:
return function(a, b) {
return complement(pred, a, b).concat(complement(pred, b, a));
};
这是我们将实际使用的功能。我们向它传递两个列表,它在第一个而不是第二个中找到元素,然后在第二个中而不是第一个中找到元素,然后组合这两个列表。
不过,再看一遍,我绝对可以从您的代码中得到提示,并通过使用some
以下方法大大简化了主要功能:
var makeSymmDiffFunc = (function() {
var complement = function(pred, a, b) {
return a.filter(function(x) {
return !b.some(function(y) {return pred(x, y);});
});
};
return function(pred) {
return function(a, b) {
return complement(pred, a, b).concat(complement(pred, b, a));
};
};
}());
complement
使用谓词并返回其第一个列表的元素,而不是第二个列表中的元素。这比我第一次使用单独的contains
函数更简单。
最后, main 函数被包装在一个立即调用的函数表达式 ( IIFE ) 中,以将内部complement
函数保持在全局范围之外。
几年后更新
现在 ES2015 已经变得非常普遍,我建议使用相同的技术,但样板文件要少得多:
const diffBy = (pred) => (a, b) => a.filter(x => !b.some(y => pred(x, y)))
const makeSymmDiffFunc = (pred) => (a, b) => diffBy(pred)(a, b).concat(diffBy(pred)(b, a))
const myDiff = makeSymmDiffFunc((x, y) => x.value === y.value && x.display === y.display)
const result = myDiff(a, b)
//=> {value="a63a6f77-c637-454e-abf2-dfb9b543af6c", display="Ryan"}