pair <int,int>对作为unordered_map问题的关键

icn*_*icn 17 c++ unordered-map std-pair

我的代码:

 typedef pair<int,int> Pair
  tr1::unordered_map<Pair,bool> h;
  h.insert(make_pair(Pair(0,0),true));
Run Code Online (Sandbox Code Playgroud)

Erorr

 undefined reference to `std::tr1::hash<std::pair<int, int> >::operator()(std::pair<int, int>) const'
Run Code Online (Sandbox Code Playgroud)

我需要修理什么?

谢谢

Mur*_*los 23

这是因为没有专门针对std::tr1::hash<Key>Key = std::pair<int, int>.你必须在宣布之前专注std::tr1::hash<Key>于.发生这种情况是因为不知道如何散列a .Key = std::pair<int, int>tr1::unordered_map<Pair,bool> h;stdpair<int, int>

以下是一个如何专业化的例子 std::tr1::hash<>

template <>
struct std::tr1::hash<std::pair<int, int> > {
public:
        size_t operator()(std::pair<int, int> x) const throw() {
             size_t h = SOMETHING;//something with x   
             return h;
        }
};
Run Code Online (Sandbox Code Playgroud)

  • @Murilo:如果没有伤害,那就不是C++. (19认同)
  • 这是不幸的,因为如果我专门在我的库中使用它,并且你专门用于你的库,并且我们的定义不相同,那么当我们的库链接在一起时,我们得到未定义的行为.`std :: tr1 :: hash`有点不足,如果可能的话,最好将`unordered_map`指定为自定义Hash类,作为第三个模板参数. (15认同)
  • @Murilo:当然有可能:`template <typename T,U> struct hash <pair <T,U >>> {size_t operator(){return hash <T>()(first)^ hash <U>()(第二); }};`.这为`pair <T,U>`提供了有效的哈希码,前提是T和U的哈希码是有效的.正如我所说,C++ 0x具有向量的通用哈希,Java和Python也用于集合.所以我不明白为什么TR1中缺少对的泛型散列,更不用说C++ 0x了. (12认同)
  • 我在ubuntu上用g ++ 4.7.3编译.在代码中粘贴代码,我得到一个错误:在不同的命名空间中模板<class_Tp> std :: tr1 :: hash的特化.所以我将代码包装在命名空间std {namespace tr1 //// code block ///}}中.然后我得到一个错误:非模板std :: tr1 <anonymous struct>的显式特化 (2认同)

fig*_*lub 6

无序映射不包含对的哈希函数,因此如果我们想要对对进行哈希处理,那么我们必须显式地为其提供一个可以对对进行哈希处理的哈希函数。

如果我们想使用pair作为unordered_map的键,有两种方法可以实现:

  1. 定义 std::hash 的专业化
typedef std::pair<std::string,std::string> pair;

struct pair_hash
{
    template <class T1, class T2>
    std::size_t operator() (const std::pair<T1, T2> &pair) const
    {
        return std::hash<T1>()(pair.first) ^ std::hash<T2>()(pair.second);
    }
};

int main()
{
    std::unordered_map<pair,int,pair_hash> unordered_map =
    {
        {{"C++", "C++11"}, 2011},
        {{"C++", "C++14"}, 2014},
        {{"C++", "C++17"}, 2017},
        {{"Java", "Java 7"}, 2011},
        {{"Java", "Java 8"}, 2014},
        {{"Java", "Java 9"}, 2017}
    };

    for (auto const &entry: unordered_map)
    {
        auto key_pair = entry.first;
        std::cout << "{" << key_pair.first << "," << key_pair.second << "}, "
                  << entry.second << '\n';
    }

    return 0;
}
Run Code Online (Sandbox Code Playgroud)
  1. 使用 Boost 库 另一个好方法是使用 Boost.function 中的 boost::hash,它可用于散列整数、浮点数、指针、字符串、数组、对和 STL 容器。
#include <iostream>
#include <boost/functional/hash.hpp>
#include <unordered_map>
#include <utility>

typedef std::pair<std::string,std::string> pair;

int main()
{
    std::unordered_map<pair,int,boost::hash<pair>> unordered_map =
    {
        {{"C++", "C++11"}, 2011},
        {{"C++", "C++14"}, 2014},
        {{"C++", "C++17"}, 2017},
        {{"Java", "Java 7"}, 2011},
        {{"Java", "Java 8"}, 2014},
        {{"Java", "Java 9"}, 2017}
    };

    for (auto const &entry: unordered_map)
    {
        auto key_pair = entry.first;
        std::cout << "{" << key_pair.first << "," << key_pair.second << "}, "
                  << entry.second << '\n';
    }

    return 0;
}
Run Code Online (Sandbox Code Playgroud)