Мой полный код слишком длинный, но вот фрагмент, который отражает суть моей проблемы:
class BPCFGParser {
public:
...
...
class Edge {
...
...
};
class ActiveEquivClass {
...
...
};
class PassiveEquivClass {
...
...
};
struct EqActiveEquivClass {
...
...
};
struct EqPassiveEquivClass {
...
...
};
unordered_map<ActiveEquivClass, Edge *, hash<ActiveEquivClass>, EqActiveEquivClass> discovered_active_edges;
unordered_map<PassiveEquivClass, Edge *, hash<PassiveEquivClass>, EqPassiveEquivClass> discovered_passive_edges;
};
namespace std {
template <>
class hash<BPCFGParser::ActiveEquivClass>
{
public:
size_t operator()(const BPCFGParser::ActiveEquivClass & aec) const {
}
};
template <>
class hash<BPCFGParser::PassiveEquivClass>
{
public:
size_t operator()(const BPCFGParser::PassiveEquivClass & pec) const {
}
};
}
Когда я компилирую этот код, я получаю следующие ошибки:
In file included from BPCFGParser.cpp:3,
from experiments.cpp:2:
BPCFGParser.h:408: error: specialization of ‘std::hash<BPCFGParser::ActiveEquivClass>’ after instantiation
BPCFGParser.h:408: error: redefinition of ‘class std::hash<BPCFGParser::ActiveEquivClass>’
/usr/include/c++/4.3/tr1_impl/functional_hash.h:44: error: previous definition of ‘class std::hash<BPCFGParser::ActiveEquivClass>’
BPCFGParser.h:445: error: specialization of ‘std::hash<BPCFGParser::PassiveEquivClass>’ after instantiation
BPCFGParser.h:445: error: redefinition of ‘class std::hash<BPCFGParser::PassiveEquivClass>’
/usr/include/c++/4.3/tr1_impl/functional_hash.h:44: error: previous definition of ‘class std::hash<BPCFGParser::PassiveEquivClass>’
Теперь мне нужно специализировать std :: hash для этих классов (поскольку стандартное определение std :: hash не включает типы, определяемые пользователем). Когда я перемещаю эти специализации шаблона до определения класса BPCFGParser
, я получаю множество ошибок из-за множества разных попыток и где-то (http://www.parashift.com/c++-faq-lite/misc-technical-issues.html) Я прочитал это:
Всякий раз, когда вы используете класс в качестве параметра шаблона, объявление этого класса должно быть полным, а не просто объявленным вперед.
Так что я застрял. Я не могу специализировать шаблоны после BPCFGParser
определения, я не могу специализировать их до BPCFGParser
определения, как я могу заставить это работать?
Вам нужно переместить специализацию во внутренний класс внутри BPCFGParser. Это соответствует обоим требованиям.
Большое спасибо за ответ :)
hash
класс определен в пространстве имен std
. Это не позволяет мне специализировать шаблоны для hash
в области, не связанной с пространством имен. Даже следующее:
template <>
class std::hash<ActiveEquivClass> {
...
не работал. Однако когда я заключаю специализации в namespace std {}
, появляется странная ошибка:
In file included from BPCFGParser.cpp:3,
from experiments.cpp:2:
BPCFGParser.h:225: error: expected unqualified-id before ‘namespace’
experiments.cpp:7: error: expected `}' at end of input
BPCFGParser.h:222: error: expected unqualified-id at end of input
В ответе, приведенном в velocityreviews, кто-то утверждает, что пространства имен не могут быть определены внутри классов. Так что я все еще застрял.