我想声明:
std::unordered_map<CString, CString> m_mapMyMap;
当我进行构建时,会出现错误提示,告诉我标准的C++不为CString提供哈希函数,而CString有(LPCSTR)运算符。
我应该如何正确实现一个适用于CString的哈希函数?
std::unordered_map<CString, CString> m_mapMyMap;
当我进行构建时,会出现错误提示,告诉我标准的C++不为CString提供哈希函数,而CString有(LPCSTR)运算符。
我应该如何正确实现一个适用于CString的哈希函数?
namespace std
{
template <typename BaseType, class StringTraits>
struct hash<CStringT<BaseType, StringTraits>>
{ // hash functor for CStringT<BaseType, StringTraits>
size_t operator()(const CStringT<BaseType, StringTraits>& _Keyval) const noexcept
{ // hash _Keyval to size_t value by pseudorandomizing transform
return (_Hash_array_representation(_Keyval.GetString(), _Keyval.GetLength()));
}
};
} // namespace std
// Usage:
std::unordered_set<CString> set1;
std::unordered_map<CString, CString> map1;
std::unordered_set<CStringA> set2;
std::unordered_map<CStringA, CStringA> map2;
std
命名空间),请使用以下代码:struct CStringHasher
{
template <typename BaseType, class StringTraits>
size_t operator()(const CStringT<BaseType, StringTraits>& _Keyval) const noexcept
{ // hash _Keyval to size_t value by pseudorandomizing transform
return std::_Hash_array_representation(_Keyval.GetString(), _Keyval.GetLength());
}
};
// Usage:
std::unordered_set<CString, CStringHasher> set1;
std::unordered_map<CString, CString, CStringHasher> map1;
std::unordered_set<CStringA, CStringHasher> set2;
std::unordered_map<CStringA, CStringA, CStringHasher> map2;
unary_function
和std::_HashSeq
)。namespace std {
template <>
struct hash<CString>
{ // hash functor for CString
size_t operator()(const CString& _Keyval) const
{ // hash _Keyval to size_t value by pseudorandomizing transform
return (_Hash_seq((const unsigned char*)(LPCWSTR)_Keyval, _Keyval.GetLength() * sizeof(wchar_t)));
}
};
template <>
struct hash<CStringA>
{ // hash functor for CStringA
size_t operator()(const CStringA& _Keyval) const
{ // hash _Keyval to size_t value by pseudorandomizing transform
return (_Hash_seq((const unsigned char*)(LPCSTR)_Keyval, _Keyval.GetLength() * sizeof(char)));
}
};
}
namespace std {
template<typename BaseType, class StringTraits>
struct hash<CStringT<BaseType, StringTraits>> : public unary_function<CStringT<BaseType, StringTraits>, size_t>
{ // hash functor for CStringT<BaseType, StringTraits>
typedef CStringT<BaseType, StringTraits> _Kty;
size_t operator()(const _Kty& _Keyval) const
{ // hash _Keyval to size_t value by pseudorandomizing transform
return (_Hash_seq((const unsigned char*)(StringTraits::PCXSTR)_Keyval,
_Keyval.GetLength() * sizeof(BaseType)));
}
};
}
std::unordered_map使用std::hash<>而不使用(LPCSTR)
运算符。
您需要重新定义哈希函数:
template<class T> class MyHash;
template<>
class MyHash<CString> {
public:
size_t operator()(const CString &s) const
{
return std::hash<std::string>()( (LPCSTR)s );
}
};
std::unordered_map<CString,CString,MyHash> m_mapMyMap;
但为了更好的性能,请使用std::string而不是CString作为键。
在尝试了MrTux的建议之后,我不得不说这已经不再起作用了。std::_HashSeq
在C++17中被移除,std::unary_function
也被移除了。
最终,我采用了另一种解决方案,结合了Microsoft关于哈希实现的建议,使用了std::basic_string_view
的哈希函数:
namespace std
{
template<typename BaseType, class StringTraits>
struct hash<ATL::CStringT<BaseType, StringTraits>>
{
size_t operator()(const ATL::CStringT<BaseType, StringTraits>& key) const noexcept
{
return hash<basic_string_view<BaseType>>()(
basic_string_view<BaseType>(
key.GetString(),
key.GetLength()));
}
};
}
std
命名空间。 - MrTuxstd
命名空间中是完全合法的。在我看来,要求额外的模板参数有点奇怪。 - Oliver Oldstd
命名空间中是完全合法的。在我看来,要求额外的模板参数有些奇怪。 - undefined首先定义这个:
struct KeyHasher
{
std::size_t operator()(const CString& k) const
{
using std::hash;
using std::string;
return hash<string>()(string(CT2CA(k)));
}
};
然后将其用作哈希函数
std::unordered_map<CString, CString, KeyHasher> umap;
CString
都必须在转换成 std::string
之前... - MrTuxCString
都必须在...之前转换为 std::string
。 - undefined