Не могу использовать макроопределение класса в C ++ - PullRequest
3 голосов
/ 30 января 2012

Я хочу сгенерировать множество подклассов, которые имеют мало различий, поэтому я хочу использовать макрос, чтобы упростить мою работу. Макрос определить ниже:

#define DECLARE_SUB_CLASS(sub_class_name, base_class_name, value1) \
class sub_class_name:base_class_name \
{ \
public: \
    virtual int initialize(const void *); \
    virtual int run(const void *); \
    virtual void reset(); \
    virtual int output(const char*); \
    virtual void terminate(); \
private: \
    static const char m_szValue=#value1; \
};

Я использую это так:

DECLARE_SUB_CLASS(RTCount13, RTCountBase, 13);

когда я компилирую с VC2005, там написано

error C2065: 'RTCount13' : undeclared identifier

в чем проблема?

Ответы [ 3 ]

8 голосов
/ 30 января 2012

Использовать gcc -E (или аналогичный для других препроцессоров)

gcc -E prepro.cxx

# 1 "prepro.cxx"
# 1 "<built-in>"
# 1 "<command-line>"
# 1 "prepro.cxx"
# 17 "prepro.cxx"
class RTCount13:RTCountBase { 
  public: 
  virtual int initialize(const void *); 
  virtual int run(const void *); 
  virtual void reset(); 
  virtual int output(const char*); 
  virtual void terminate(); 
  private: 
  static const char m_szValue="13"; 
};;

Вы пытаетесь присвоить "13" символу.

Кстати, вы также можете использовать шаблон вместо макроса, чтобы делать то же самое, что и ваш макрос (а именно, объявлять, но не определять методы). Вот полный (урезанный) пример с отдельными определениями методов.

#include <iostream>
class RTCountBase {};

template <class base_class_name, int v>
class RTCount: base_class_name { 
  public: 
   virtual int output(); 
   virtual void terminate(); 
  private: 
   static const int m_szValue=v; 
};

template <class base_class_name, int v>
int RTCount<base_class_name,v>::output(){ return m_szValue; }

template <class base_class_name, int v>
void RTCount<base_class_name,v>::terminate(){ std::cout <<" term "<<std::endl; }

typedef RTCount<RTCountBase,13> RTCount13; // typedef instead of macro
typedef RTCount<RTCountBase,14> RTCount14;

int main(){
   RTCount13 myc13;
   RTCount14 myc14;
   std::cout << "my13: "<<myc13.output()<<std::endl;
   std::cout << "my14: "<<myc14.output()<<std::endl;
   return 0;
}
2 голосов
/ 30 января 2012

Поскольку вы используете C ++, вам действительно следует избегать использования #define для этого. В языке C ++ есть лучшие способы сделать это, например:

template <class B, const char C>
class SomeClass : public B
{
public:
  virtual int initialize(const void *) {return m_szValue;}
  virtual int run(const void *) {return 0;}
  virtual void reset() {}
  virtual int output(const char*) {return 0;}
  virtual void terminate() {}
private: 
  static const char m_szValue=C; 
};

и затем измените это:

DECLARE_SUB_CLASS(RTCount13, RTCountBase, 13);

RTCount13 some_instance;

до:

SomeClass <RTCountBase, 13> some_instance;
0 голосов
/ 30 января 2012

При определении статического символа у вас отсутствует знак хеша:

static const char m_szValue=#value1;

вместо

static const char m_szValue=##value1;

EDIT:

Если m_szValue предназначено для string, вам нужно определить его вне объявления класса. Я предлагаю вам определить макрос для этого также, чтобы сохранить последовательность:

#define DECLARE_SUB_CLASS(sub_class_name, base_class_name) \
class sub_class_name:base_class_name \
{ \
    //.....\
private: \
    static const char* m_szValue; \
};

#define DEFINE_SUB_CLASS_STATIC(sub_class_name, value1) \
    const char* sub_class_name::m_szValue = #value1; \

который вы используете так:

//header file
DECLARE_SUB_CLASS(RTCount13, RTCountBase);

//impl file:
DEFINE_SUB_CLASS_STATIC(RTCount13, 13);
...