首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >如何为QSet<SomeClass*>容器编写qHash?

如何为QSet<SomeClass*>容器编写qHash?
EN

Stack Overflow用户
提问于 2012-02-03 17:35:17
回答 2查看 5.6K关注 0票数 9

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

代码如下:

代码语言:javascript
复制
    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*>

为了能够使用QSet< QSet<SomeClass*> >,我如何实现qHash(QSet<Custom*>)

编辑:

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

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2012-02-03 17:41:32

对容器进行散列的一种常见方法是组合所有元素的散列。Boost为此提供了hash_combine and hash_range。这应该会让您了解如何为qHash的结果实现此功能。

因此,给定CustomqHash

代码语言:javascript
复制
uint qHash(const QSet<Custom*>& c) {
  uint seed = 0;

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

  return seed;
}
票数 4
EN

Stack Overflow用户

发布于 2014-03-14 18:36:50

你不能用boost::hash_range/boost::hash_combine实现qHash (这是pmr的答案,有效地做到了这一点),因为QSetstd::unordered_set的Qt等价物,并且,顾名思义,这些容器是无序的,而Boost Documentation声明hash_combine是依赖于顺序的,即。它会将排列散列为不同的散列值。

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

代码语言:javascript
复制
For all x, y:   x == y => qHash(x) == qHash(y)

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

代码语言:javascript
复制
template <typename T>
inline uint qHash(const QSet<T> &set, uint seed=0) {
    return std::accumulate(set.begin(), set.end(), seed,
                           [](uint seed, const T&value) {
                               return seed + qHash(value); // or ^
                           });
}
票数 5
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/9126470

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档