使用 RxJs groupBy 以对象作为键

Using RxJs groupBy with objects as keys

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

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

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

当然有一个关于对象相等性的问题,但这是我感到困惑的地方,因为我不明白如何使用 groupBy 的附加参数。最新版本的文档说有第三个参数可以是 comparer,但它永远不会被调用。较早的文档讨论了 key serializer 这是一个完全不同的想法,但这两种方法都不适合我。

查看 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?

是我想写的,但运气不好,我为第三次回调所做的任何事情都没有被调用。

这是怎么回事?

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

hashCodevalueOf 应该类似于 c# Object.GetHashCode.

他们应该 return 一个这样的值:

  • 两个被认为相等的键每次都应该return相同的值。
  • 被认为不相等的两个键通常应该 return 不同的值(以尽量减少冲突)。你越能确保这一点,字典的效率就会越高。

不同之处在于 hashCode 应该 return 一个数字,而 valueOf 可以 return 一个数字或字符串。因此 valueOf 更容易实现。

comparer 的规则是它需要 2 个键值,并且应该 return 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; });