Specialize hashmap template
I'm using the hash_map in C++ and want to supply a simplified type name for it:
The key type and hasher function are always the same.
stdext::hash_map<std::string, [MAPPED TYPE], CStringHasher>
However, I don't really want to write all this every time I declare a hash map which maps strings to type X.
- Can I write a template or macro that simplifies this?
So the above declaration would look like:
template<typename T> StringHashmap = stdext::hash_map<std::string, T, CStringHasher>
StringHashmap<int> mapA; //Maps std::string 开发者_运维知识库to int
StringHashamp<bool> mapB; //Maps std::string to bool
As others said, template aliases is the way to go if you can use C++0x:
template < typename MappedType >
using StringHashMap = stdext::hash_map< std::string, MappedType, CStringHasher >;
StringHashMap< int > mapA;
StringHashMap< bool > mapB;
(As @MSalters noted, if you have C++0x available, you should probably use std::unordered_map
.)
Otherwise, you can still use the usual workaround, which is to define a class template containing a typedef:
template < typename MappedType >
struct StringHashMap
{
typedef stdext::hash_map< std::string, MappedType, CStringHasher > type;
};
StringHashMap< int >::type mapA;
StringHashMap< bool >::type mapB;
A drawback with this solution (which have engendered many questions here on SO) is that you sometimes need to prefix StringHashMap< T >::type
by the typename
keyword in order to assert to the compiler that this name effectively denotes a type. I will not dwell on this subject in this answer, you can check out this FAQ, and especially the accepted answer, for more information. (Thanks to @sbi for evoking this matter.)
With the current version of C++ (C++03), there really isn't a good way to do this. You could try something like
#define StringHashMap(type) stdext::hash_map<std::string, type, CStringHasher>
But then you run into trouble if you try to specify a template type with a comma in it, like this:
StringHashMap(pair<int, string>) myMap; // Error!
This fails because the preprocessor will tokenize this as
StringHashMap((pair<int), (string>)) myMap; // Error!
which isn't what you want. If you're not going to do anything like this, though, you should be fine. You could alternatively use typedef
s to get around it:
typedef pair<int, int> IntPair;
StringHashMap(IntPair) myMap; // Okay
If you're allowing use of C++0x features, you can use a template using declaration, like this one here:
template <typename T>
using StringHashMap = stdext::hash_map<std::string, T, CStringHasher>;
Unfortunately, C++03 doesn't have a "template typedef" feature like this one, so you'd have to use a newer compiler and won't have as good a portability guarantee.
Hey, wait a minute! My name is templatetypedef
and this is the first time on Stack Overflow that I've ever written that phrase in an answer! You just made my night. :-)
If you're using C++ 0x, template typedefs were designed to do exactly this kind of thing:
http://en.wikipedia.org/wiki/C%2B%2B0x#Template_aliases
精彩评论