使用lodash按键过滤多个值

12

我正在尝试在 lodash 这个 JavaScript 库中查找如何筛选数组中多个键的值。类似于 SQL 中的 WHERE KEY in (val1, val2)。

鉴于此,以下是一个示例:

var users = [
   { 'user''barney''age'36'active'true },
   { 'user''fred',   'age'40'active'false },
   { 'user''Avding',   'age'34'active'true }
];

_.filter(...)

如何找到年龄在 (34, 36) 范围内的用户? 数字也可能在运行时发生变化。

5个回答

8

Lodash的filter接受一个谓词。您可以使用部分应用程序创建谓词,以便轻松更改值:

var users = [
   { 'user': 'barney', 'age': 36, 'active': true },
   { 'user': 'fred',   'age': 40, 'active': false },
   { 'user': 'Avding',   'age': 34, 'active': true }
];

var predicate = function(start, end) {
  return function(user) {
    return _.inRange(user.age, start, end + 1);
  }
}

var result = _.filter(users, predicate(34, 36));

console.log('inRange', result);

/** is equal predicate **/

var predicate = function() {
  var args = _.toArray(arguments);
  
  return function(user) {
    var equalToUserAge = _.partial(_.isEqual, user.age);
    
    return args.some(equalToUserAge);
  }
}

var result = _.filter(users, predicate(34, 40));

console.log('equals', result);
<script src="https://cdnjs.cloudflare.com/ajax/libs/lodash.js/4.16.4/lodash.min.js"></script>

您也可以使用原生JS轻松实现这一功能(示例使用ES6):

var users = [
   { 'user': 'barney', 'age': 36, 'active': true },
   { 'user': 'fred',   'age': 40, 'active': false },
   { 'user': 'Avding',   'age': 34, 'active': true }
];

const predicate1 = (start, end) => ({ age }) =>
  age >= start && age <= end;

const result1 = users.filter(predicate1(34, 36));

console.log('in range', result1);

const predicate2 = (...args) => ({ age }) =>
  args.some((num) => age === num);

const result2 = users.filter(predicate2(34, 40));

console.log('is equal', result2);


1
实际上,您的建议可以作为“中间值”使用,但是我需要精确匹配,因此它应该返回age == 34或age == 36。如果我需要一些东西(34,40),怎么办?您的代码返回3个对象,但我只需要看到年龄为34和40的2个对象。 - Avi Kenjale
将谓词更改为您喜欢的任何内容。请参见代码中的示例谓词。 - Ori Drori

4
const users = [
  { 'user': 'barney', 'age': 36, 'active': true },
  { 'user': 'fred',   'age': 40, 'active': false },
  { 'user': 'Avding',   'age': 34, 'active': true }
];

const minAge = 34;
const maxAge = 36;

const filtered = _.filter(users, (user) => {
  return user.age >= minAge && user.age <= maxAge;
});

0

var users = [
  { user: 'barney', age: 36, active: true },
  { user: 'fred',  age: 40, active: false },
  { user: 'travis', age: 34, active: true}
];
var result = []
var x = _.filter(users, function(o) {
  _.map([36,34], function(p) {
    if (o.age == p) {
      result.push(o);
    }
  });
});
console.log(result)
<script src="https://cdnjs.cloudflare.com/ajax/libs/lodash.js/4.17.21/lodash.min.js"></script>


0

虽然这是老问题,但我也遇到了同样的问题。我不知道它是否高效,但下面的函数返回了我需要的内容。我必须将结果展开以保持与输入相同的格式。

var users = [
    { 'user': 'barney', 'age': 36, 'active': true },
    { 'user': 'fred',   'age': 40, 'active': false },
    { 'user': 'Avding',   'age': 34, 'active': true }
];

var ages = [{"age": 36},{"age": 34}]

function filterAges(array, vals) {
    var result=[];
    for (i = 0; i < vals.length; i++) {
        result.push(_.filter( array, vals[i] ));
    }
    return _.flattenDepth(result, 1);
}

-5
你可以像这样做:
_.filter( object, { 'age': '34', 'age': '24' } );

问题在于,如果您正在动态创建此内容,则无法在对象中多次使用相同的键。

3
这不能工作的原因是,一个对象只能有一个键,将 'age' 设定两次意味着在地图中只会显示最后一个(24)。这意味着你只会过滤出年龄为 24 的结果,而不是年龄为 34 的结果。 - Patrick Connelly

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接