如何为QSet <someclass *>容器编写qHash?</someclass *>

时间:2012-02-03 09:35:17

标签: c++ qt hash containers

我需要在我的应用程序中实现一组集合。 将QSet与自定义类一起使用需要提供qHash()函数和operator==

代码如下:

    class Custom{
        int x;
        int y;
        //some other irrelevant here
    }
    inline uint qHash(Custom* c){
        return (qHash(c->x) ^ qHash(c->y));
    }
    bool operator==(Custom &c1, Custom &c2){
        return ((c1.x==c2.x) && (c1.y == c2.y));
    }

    //now I can use: QSet<Custom*>

如何实施qHash(QSet<Custom*>),以便能够使用QSet< QSet<SomeClass*> >

编辑:

其他问题: 在我的应用程序中,“集合集”最多可包含15000套。每个子集最多25个Custom类指针。如何保证qHash(QSet<Custom*>)足够独特?

2 个答案:

答案 0 :(得分:5)

您无法使用qHash / boost::hash_combine实现boost::hash_range(这是pmr的回答有效),因为QSet是Qt等效于{ {1}},并且,正如STL名称所示,这些容器是无序,而Boost Documentation表示std::unordered_set依赖于顺序,即。它会将排列方式哈希到不同的哈希值。

这是一个问题,因为如果您天真地按存储顺序散列组合元素 你无法保证两个比较相等的集合确实相等,这是散列函数的要求之一:

hash_combine

因此,如果您的哈希组合函数需要为输入值的任何排列生成相同的输出,则它需要可交换。幸运的是,两个(无符号)加法和xor操作都符合要求:

For all x, y:   x == y => qHash(x) == qHash(y)

答案 1 :(得分:4)

散列容器的常用方法是组合所有元素的散列。为此,Boost提供hash_combine and hash_range。这可以让您了解如何针对qHash

的结果实施此功能

所以,考虑到qHash Custom

uint qHash(const QSet<Custom*>& c) {
  uint seed = 0;

  for(auto x : c) {
    seed ^= qHash(x) + 0x9e3779b9 + (seed << 6) + (seed >> 2);
  }

  return seed;
}