1

ramda中,我们得到了很多高阶组合器来做实际工作。一个非常方便的是.filter()实用程序——我们Array.filter()在 ES6 中得到了这个函数,但是 ramda(我相信还有 lodash)也给了我们过滤对象的能力。对我来说,我想要一个工具来同时过滤键和值。

ES6 中的const关键字是一个非常强大的工具 - 通过将它与任意(非对象)值一起使用,我们可以保证该值在块的其余部分永远不会改变(并帮助我们推断错误,许多其他非常不错的收获)。从对象中过滤键和值的唯一其他实用方法是 using for .. in,它会强制您使用可变变量:

let xs = {...}
for (const k in xs) {
  if (someProperty(k,xs[k])) {
    delete xs[k]
  }
}

我认为使用filterWithKey可以缓解这个问题:

const xs = filterWithKey(someProperty, {...})

在流行的 javascript 库中有类似的东西吗?我还没有在 ramda 或 lodash 中看到过类似的东西,我不知道还能去哪里看。我认为这将是一个合适的实现:

function filterWithKey (p,xs) {
  R.toPairs(xs).reduce((acc,x) => {
    const k = x[0]
    const v = x[1]
    return p(k,v) ? acc[k] = v : acc
  })
}

它仍然需要重建对象的行为。有更好的解决方案吗?

4

1 回答 1

1

Ramda 最近才添加了根据对象的值过滤对象的功能;在那之前,它本机支持列表,否则将委托给对象的filter方法(如果存在)。但在第 1429 期中,它也被扩展到普通对象。

在那个问题上也有很多关于支持keys参数的可能性的讨论,但最终证明它与库的其他部分不一致。

我刚刚在Ramda 的 Cookbook中添加了一个部分,描述了为自己编写此函数的一种方法:

const filterWithKeys = (pred, obj) => R.pipe(
  R.toPairs, 
  R.filter(R.apply(pred)), 
  R.fromPairs
)(obj);

filterWithKeys(
  (key, val) => key.length === val, 
  {red: 3, blue: 5, green: 5, yellow: 2}
); //=> {red: 3, green: 5}

如果您有更有用的简短示例,我很乐意将其包含在内。

就像 Ramda 世界中的任何东西一样,它不会改变您的输入数据。这在函数式编程中非常重要,我对使用这种方式构建的版本不感兴趣。换句话说,我认为这是一个功能:

它仍然需要重建对象的行为。

于 2016-01-17T01:33:15.680 回答