lodash:使用不同的对象数组过滤对象数组

mg1*_*075 12 javascript arrays lodash

这个问题是针对lodash的.

给定两个对象数组,使用另一个数组的对象过滤一个数组的最佳方法是什么?我试图在下面提出一个场景,我这样做的方式是使用两个.forEach循环,但我想知道如果使用lodash,有更好的方法来进行这种类型的过滤.


示例
对象的主要源数组是users.

var users = [
  { 'user': 'barney', 'age': 36, 'active': true },
  { 'user': 'joe', 'age': 40, 'active': false },
  { 'user': 'fred', 'age': 50, 'active': false },
  { 'user': 'fred', 'age': 60, 'active': false },
  { 'user': 'fred', 'age': 70, 'active': false },
  { 'user': 'fred', 'age': 22, 'active': false },
  { 'user': 'fred', 'age': 25, 'active': false },
  { 'user': 'barney', 'age': 40, 'active': false },
  { 'user': 'pebbles', 'age': 1,  'active': true }
];
Run Code Online (Sandbox Code Playgroud)

users调用将过滤数组的对象数组others.

var others = [
  { 'user': 'fred', 'age': 60 },
  { 'user': 'fred', 'age': 70},
  { 'user': 'fred', 'age': 22}
];
Run Code Online (Sandbox Code Playgroud)

基于others过滤的期望结果users是:

[
  { 'user': 'fred', 'age': 60, 'active': false },
  { 'user': 'fred', 'age': 70, 'active': false },
  { 'user': 'fred', 'age': 22, 'active': false }
];
Run Code Online (Sandbox Code Playgroud)

这是获得所需结果的一种方法.

var result = [];

_.forEach(users, function (n, key) {
   _.forEach(others, function (n2, key2) {
      if (n.user === n2.user && n.age === n2.age) {
         result.push(n);
      }
   });
});

console.log(result);
Run Code Online (Sandbox Code Playgroud)

以下是jsbin的示例.
http://jsbin.com/hapariviya/1/edit?html,js,console,output

Cha*_*vov 9

这是我能想到的清洁方式:

var result = _.flatten(_.map(others, function(item){
  return _.filter(users, item);
}));
Run Code Online (Sandbox Code Playgroud)

编辑: 道歉JS Bin输出混淆了嵌套数组.


Cor*_*son 8

您可以为其他人编制索引,然后在不必嵌套循环的情况下获得所需的结果.无论数据量多少,它都应该是一个相对有效的解决方案:

// index others by "user + age"
var lookup = _.keyBy(others, function(o) { return o.user + o.age.toString() });
// find all users where "user + age" exists in index, one loop, quick lookup. no nested loops
var result = _.filter(users, function(u) {
    return lookup[u.user + u.age.toString()] !== undefined;
});
Run Code Online (Sandbox Code Playgroud)

这给出了相同的结果:

[
  { 'user': 'fred', 'age': 60, 'active': false },
  { 'user': 'fred', 'age': 70, 'active': false },
  { 'user': 'fred', 'age': 22, 'active': false }
];
Run Code Online (Sandbox Code Playgroud)

有趣的是,您的原始解决方案是所有这些答案中最高效的解决方案.

http://jsperf.com/testingdiwq

这里的性能问题几乎可以忽略不计.在大多数情况下,DOM交互是前端的主要性能瓶颈.如果你是针对巨大的数据集运行它并注意到锁定,你肯定希望通过使用for循环而不是使用lodash函数进行迭代来进一步优化它....但是你通常不会遇到这种类型的数据JavaScript ...... SQL和其他人会更好地处理它.


小智 5

使用ES6粗箭头和lodash拒绝:

const result = _.reject(users, (item) => _.find(others, { user: item.user }));
Run Code Online (Sandbox Code Playgroud)