Фабрика классов C ++ 0x с проблемой вариационных шаблонов - PullRequest
7 голосов
/ 25 мая 2010

У меня есть фабрика классов, где я использую шаблоны переменных для параметров c'tor (код ниже). Однако, когда я пытаюсь использовать его, я получаю ошибки компиляции; когда я изначально писал без параметров, все работало нормально.

Вот класс:

template< class Base, typename KeyType, class... Args >
class GenericFactory
{
public:
   GenericFactory(const GenericFactory&) = delete;
   GenericFactory &operator=(const GenericFactory&) = delete;

   typedef Base* (*FactFunType)(Args...);

   template <class Derived>
   static void Register(const KeyType &key, FactFunType fn)
   {
      FnList[key] = fn;
   }

   static Base* Create(const KeyType &key, Args... args)
   {
      auto iter = FnList.find(key);
      if (iter == FnList.end())
         return 0;
      else
         return (iter->second)(args...);
   }

   static GenericFactory &Instance() { static GenericFactory gf; return gf; }
private:
   GenericFactory() = default;

   typedef std::unordered_map<KeyType, FactFunType> FnMap;
   static FnMap FnList;
};

template <class B, class D, typename KeyType, class... Args>
class RegisterClass
{
public:
   RegisterClass(const KeyType &key)
   {
      GenericFactory<B, KeyType, Args...>::Instance().Register(key, FactFn);
   }
   static B *FactFn(Args... args)
   {
      return new D(args...);
   }
};

Вот ошибка: при звонке (например,)

// Tucked out of the way
RegisterClass<DataMap, PDColumnMap, int, void *> RC_CT_PD(0);

GCC 4.5.0 дает мне:

In constructor 'RegisterClass<B, D, KeyType, Args>::RegisterClass(const KeyType&) [with B = DataMap, D = PDColumnMap, KeyType = int, Args = {void*}]':
no matching function for call to 'GenericFactory<DataMap, int, void*>::Register(const int&, DataMap* (&)(void*))'

Я не понимаю, почему он не скомпилируется, и после продолжительного поиска в Интернете я не смог найти ответ. Может кто-нибудь сказать мне, что я делаю неправильно (кроме странного имени переменной, которое имеет смысл в контексте)?

1 Ответ

2 голосов
/ 26 мая 2010

Я думаю, что это раздражает здесь:

template <class Derived> 
static void Register(const KeyType &key, FactFunType fn) 
{ 
   FnList[key] = fn; 
} 

Вы не используете Derived в этой функции, но, вероятно, это мешает попытке gcc разрешить GenericFactory<...>.Register(...). Вы также можете изменить это значение на GenericFactory<...>::Register(...).

...