使用 RxJs groupBy 将对象作为键

Using RxJs groupBy with objects as keys

本文关键字:对象 RxJs groupBy 使用      更新时间:2023-09-26

我正在尝试将groupBy与RxJs一起使用,我需要将对象用作键。如果我不这样做,例如,我使用像这样的简单字符串:

var types = stream.groupBy(
    function (e) { return e.x; }); //x is a string

然后一切顺利,我的订阅被调用一次,并且只调用一次,用于每个不同的密钥。但是如果我尝试使用对象,则会为 stream 中的每个元素调用订阅,即使键恰好与以前的键相同。

当然,对象相等性存在问题,但这就是我感到困惑的地方,因为我不明白如何使用额外的参数来groupBy。最新版本的文档说有第三个参数可以作为比较器,但它从未被调用。早期的文档谈到了一个密钥序列化程序,这是一个完全不同的想法,但这两种方法都不适合我。

查看 Rx 源代码,我看到尝试检查getHashCode函数,但我没有找到和文档。像这样编写代码:

var types = stream.groupBy(
    function (e) {
        return { x: e.x, y: e.y };   //is this valid at all?
    },
    function (e) { return e; },
    function (...) { return ???; }); //what am I supposed to do here?

是我想写的,但没有运气,我为第三个回调放的任何内容都没有被调用。

这是怎么回事?

最简单的解决方案是确保键函数返回数字或字符串。 但是如果你真的想返回一个对象作为你的键,那么你可以使用comparerhashCode来帮助groupBy。 您可以使用 valueOf 来返回字符串,而不是 hashCode(这需要您返回一个数字)。

hashCodevalueOf的工作方式应该类似于 c# Object.GetHashCode。

它们应返回一个值,以便:

  • 被视为相等的两个键每次都应返回相同的值。
  • 被视为不相等的两个键通常应返回不同的值(以最大程度地减少冲突)。 您越能确保这一点,字典的效率就越高。

区别在于hashCode应该返回一个数字,而valueOf可以返回一个数字或一个字符串。 因此valueOf更容易实现。

comparer的规则是它需要 2 个键值,并应返回 true 以指示相等,false以指示不平等。

因此,您可以将示例编写为:

var valueOf = function () {
    return JSON.stringify(this);
};
var keyCompare = function (a, b) { return a.x === b.x && a.y === b.y; };
var types = stream.groupBy(
    function (e) {
        return { x: e.x, y: e.y, valueOf: valueOf };   //is this valid at all?
    },
    function (e) { return e; },
    keyCompare);

但是,如果你的valueOf函数实际上正在生成与你的比较器匹配的唯一值,并且你并不真正关心键是否作为实际对象被放置在下游,那么只需让你的生活更轻松,并将你的键转换为字符串并使用字符串键,如下所示:

var types = stream.groupBy(
    function (e) { return JSON.stringify({ x: e.x, y: e.y }); },
    function (e) { return e; });