Повышение препроцессора генерации нескольких шаблонных классов - PullRequest
2 голосов
/ 04 декабря 2011

Я пытаюсь превратить этот код C ++ / ответ (класс, который создает карту string <-> some_array_of_predefined_function_types) в более автоматизированный класс с использованием boost / препроцессора.

Я создал следующий код:

#include <boost/preprocessor.hpp>
#include <iostream>
#include <string>
#include <map>
#include <vector>

#define CF_ADD_MAPS(z, n, T) \
std::map<std::string, identity< BOOST_PP_CAT(T, n ) >::type* >  \
BOOST_PP_CAT(T , BOOST_PP_CAT(n , _var));  

#define CF_ADD_OPERATORS(z, n, T) \
operator identity< BOOST_PP_CAT(T, n ) >::type*()  \
{return \
BOOST_PP_CAT( p->   , BOOST_PP_CAT(T , BOOST_PP_CAT(n , BOOST_PP_CAT( _var, [n]) )) ) ; \
}

#define CF_ADD_INSERTERS(z, n, T) \
    void insert(std::string name, identity< BOOST_PP_CAT(T, n ) >::type* ptr)   \
{ \
    BOOST_PP_CAT(T , BOOST_PP_CAT(n , BOOST_PP_CAT( _var, [name]) )) =ptr; \
}

template <typename T> struct identity { typedef T type; };

#define CF_DEFINE_MAP_STRUCT_I(name, number)                   \
template <BOOST_PP_ENUM_PARAMS(number, class T)>              \
class name {                                          \
      BOOST_PP_REPEAT(number, CF_ADD_MAPS, T)        \
      friend class apitemp;                                        \
      public: \
      class apitemp {  \
        std::string n;  \
        BOOST_PP_CAT(name, *) p; \
        public: \
            apitemp(std::string name_, BOOST_PP_CAT(name, *) parent)  \
                : n(name_), p(parent) {} \
            BOOST_PP_REPEAT(number, CF_ADD_OPERATORS, T) \
      }; \
      BOOST_PP_REPEAT(number, CF_ADD_INSERTERS, T) \
      apitemp operator[](std::string n) {return apitemp(n, this);} \
};


#define CF_MAP_CLASS_NAME my_map
#define CF_GENERATE_MAP_CLASS(z, n, data) CF_DEFINE_MAP_STRUCT_I(CF_MAP_CLASS_NAME, n)
#define CF_MAX_MAP_TYPES_COUNT 2
BOOST_PP_REPEAT(CF_MAX_MAP_TYPES_COUNT, CF_GENERATE_MAP_CLASS, x)


/*
class api {
    //maps containing the different function pointers
    std::map<std::string, identity<void()>::type* > voida;
    std::map<std::string, identity<int(std::string, const int&)>::type* > stringcrint;
    friend class apitemp;
public:
    //api temp class 
    //given an api and a name, it converts to a function pointer  
    //depending on parameters used
    class apitemp {
        std::string n;
        api* p;
    public:
        apitemp(std::string name, api* parent) 
            : n(name), p(parent) {}
        operator identity<void()>::type*()
        {return p->voida[n];}
        operator identity<int(std::string, const int&)>::type*()
        {return p->stringcrint[n];}
    }; 
    //insertion of new functions into appropriate maps
    void insert(std::string name, identity<void()>::type* ptr) 
    {voida[name]=ptr;}
    void insert(std::string name, identity<int(std::string, const int&)>::type* ptr)
    {stringcrint[name]=ptr;}
    //operator[] for the name gets halfway to the right function
    apitemp operator[](std::string n) {return apitemp(n, this);}
} myMap;
*/

int hello_world(std::string name, const int & number )
{
    name += "!";
    std::cout << "Hello, " << name << std::endl;
    return number;
}

int main() {
    my_map<int(std::string, const int &)> myMap;
    myMap.insert("my_method_hello", &hello_world ); 
    int a = myMap["my_method_hello"]("Tim", 25);
    std::cout << a << std::endl;
    std::cin.get();
}

Что не компилируется со следующими 5 ошибками:

Error   1   error C2913: explicit specialization; 'my_map' is not a specialization of a class template  

Error   2   error C2133: 'myMap' : unknown size

Error   3   error C2512: 'my_map' : no appropriate default constructor available

Error   4   error C2039: 'insert' : is not a member of 'my_map'

Error   5   error C2676: binary '[' : 'my_map' does not define this operator or a conversion to a type acceptable to the predefined operator

в то время как препроцессор генерирует:

template <>
class my_map {
    friend class apitemp;
public:
    class apitemp {
        std::string n;
        my_map* p;
    public: apitemp(std::string name_, my_map* parent) : n(name_), p(parent) {}  
    };
    apitemp operator[](std::string n) {
        return apitemp(n, this);
    } 
};
template < class T0>
class my_map {
    std::map<std::string, identity< T0 >::type* > T0_var;
    friend class apitemp;
public:
    class apitemp {
        std::string n;
        my_map* p;
    public:
        apitemp(std::string name_, my_map* parent) : n(name_), p(parent) {} 
        operator identity< T0 >::type*() {
            return p->T0_var[0] ; 
        } 
    }; 
    void insert(std::string name, identity< T0 >::type* ptr) {
        T0_var[name] =ptr;
    } 
    apitemp operator[](std::string n) {
        return apitemp(n, this);
    } 
};

Мне интересно, что я сделал не так и как заставить его скомпилировать?

1 Ответ

1 голос
/ 04 декабря 2011
template <>
class my_map {
  // ...
};

Проблема здесь с BOOST_PP_ENUM_PARAMS, который также генерирует версию с 0 параметрами.И поскольку версии с 0 параметрами (template<>) вводят специализации, это вызывает у вас ошибки.Исправлено: Иметь хотя бы один фиксированный параметр.

Следующая проблема: Вы не можете «перегружать» классы на основе параметров шаблона.Они должны иметь разные имена или иметь другую структуру с частичной специализацией или что-то в этом роде.

...