1.應用:
也許需要一個腳本,腳本中有一個參數傳遞類名,你需要通過類名來動態的創建這個對象,這時候就需要動態創建了
2.原理:
C++不是動態語言,所以沒法從語言機制上實現類的動態創建,但這樣的需求卻有可能存在,一個類似的例子便是MFC中CWnd類的Create方法,其第一個參數為Window Class的名字,這就允許用戶通過class的名字來創建相應的窗口。
也許需要一個腳本,腳本中有一個參數傳遞類名,你需要通過類名來動態的創建這個對象,這時候就需要動態創建了
2.原理:
C++不是動態語言,所以沒法從語言機制上實現類的動態創建,但這樣的需求卻有可能存在,一個類似的例子便是MFC中CWnd類的Create方法,其第一個參數為Window Class的名字,這就允許用戶通過class的名字來創建相應的窗口。
要想實現這一點,必須有一個“管理中心”,用于登記類的名字,并且通過名字能夠調用對應的創建方法來創建相應的實例。結合類工廠的設計思想,這里我們讓一套繼承體系中的基類作為“管理中心”,由它來維護所有派生類的必要信息,包括類名和工廠函數,這二者必須建立起映射關系,map是不錯的選擇。定義了一個派生類后,它就自動向基類進行注冊
3.實現:
可以用宏來代替:
我們需要做的就是是哦那個這四個宏
附加上測試代碼:
參考地址:http://blog.csdn.net/freefalcon/article/details/109275
3.實現:
可以用宏來代替:
/**
實現原理:
基類里面用map管理所有需要動態創建 【類名-創建對象的函數指針】
基類聲明中需要添加 DECLEAR_DYNCRT_BASE 宏,cpp實現中要添加 IMPLEMENT_DYNCRT_BASE 宏
派生類聲明中需要添加 DECLEAR_DYNCRT_CLASS 宏,cpp實現中要添加 IMPLEMENT_DYNCRT_CLASS 宏
派生類宏中創建了一個全局對象向基類map中注冊【類名-創建對象的函數指針】
*/
#include <map>
// 聲明具有動態創建的基類
#define DECLEAR_DYNCRT_BASE(base) \
public: \
typedef base* (*ClassGen)(); \
static void _register(const char* class_name, ClassGen class_gen) \
{ \
class_set.insert(std::map<const char*, ClassGen>::value_type(class_name, class_gen)); \
} \
static base* create(const char* class_name) \
{ \
std::map<const char*, ClassGen>::iterator it = class_set.find(class_name); \
if (it != class_set.end()) \
return (it->second)(); \
return NULL; \
} \
protected: \
static std::map<const char*, ClassGen> class_set;
// 用于實現基類
#define IMPLEMENT_DYNCRT_BASE(base) \
std::map<const char*, base::ClassGen> base::class_set;
// 用于聲明一個能夠被動態創建的類(用一個全局對象進行注冊)
#define DECLEAR_DYNCRT_CLASS(derived, base) \
public: \
struct derived##register \
{ \
derived##register() \
{ \
static bool bRegister = false; \
if (!bRegister) \
{ \
base::_register(#derived, _CreateObjectByName); \
bRegister = true; \
} \
} \
}; \
static base* _CreateObjectByName() \
{ \
return new derived; \
}
// 用于實現一個能被動態創建的類
#define IMPLEMENT_DYNCRT_CLASS(derived) \
static derived::derived##register _##derived##global_object;
實現原理:
基類里面用map管理所有需要動態創建 【類名-創建對象的函數指針】
基類聲明中需要添加 DECLEAR_DYNCRT_BASE 宏,cpp實現中要添加 IMPLEMENT_DYNCRT_BASE 宏
派生類聲明中需要添加 DECLEAR_DYNCRT_CLASS 宏,cpp實現中要添加 IMPLEMENT_DYNCRT_CLASS 宏
派生類宏中創建了一個全局對象向基類map中注冊【類名-創建對象的函數指針】
*/
#include <map>
// 聲明具有動態創建的基類
#define DECLEAR_DYNCRT_BASE(base) \
public: \
typedef base* (*ClassGen)(); \
static void _register(const char* class_name, ClassGen class_gen) \
{ \
class_set.insert(std::map<const char*, ClassGen>::value_type(class_name, class_gen)); \
} \
static base* create(const char* class_name) \
{ \
std::map<const char*, ClassGen>::iterator it = class_set.find(class_name); \
if (it != class_set.end()) \
return (it->second)(); \
return NULL; \
} \
protected: \
static std::map<const char*, ClassGen> class_set;
// 用于實現基類
#define IMPLEMENT_DYNCRT_BASE(base) \
std::map<const char*, base::ClassGen> base::class_set;
// 用于聲明一個能夠被動態創建的類(用一個全局對象進行注冊)
#define DECLEAR_DYNCRT_CLASS(derived, base) \
public: \
struct derived##register \
{ \
derived##register() \
{ \
static bool bRegister = false; \
if (!bRegister) \
{ \
base::_register(#derived, _CreateObjectByName); \
bRegister = true; \
} \
} \
}; \
static base* _CreateObjectByName() \
{ \
return new derived; \
}
// 用于實現一個能被動態創建的類
#define IMPLEMENT_DYNCRT_CLASS(derived) \
static derived::derived##register _##derived##global_object;
我們需要做的就是是哦那個這四個宏
附加上測試代碼:
// 測試實例:
class Base
{
DECLEAR_DYNCRT_BASE(Base)
DECLEAR_DYNCRT_CLASS(Base, Base)
public:
virtual void Print()
{
std::cout << "This is base!" << std::endl;
}
};
IMPLEMENT_DYNCRT_BASE(Base)
IMPLEMENT_DYNCRT_CLASS(Base)
class Derived : public Base
{
DECLEAR_DYNCRT_CLASS(Derived, Base)
public:
virtual void Print()
{
cout << "This is derived!" << endl;
}
};
IMPLEMENT_DYNCRT_CLASS(Derived)
int main()
{
Base* base = Base::create("Base");
if (base)
base->Print();
system("pause");
return 0;
}
class Base
{
DECLEAR_DYNCRT_BASE(Base)
DECLEAR_DYNCRT_CLASS(Base, Base)
public:
virtual void Print()
{
std::cout << "This is base!" << std::endl;
}
};
IMPLEMENT_DYNCRT_BASE(Base)
IMPLEMENT_DYNCRT_CLASS(Base)
class Derived : public Base
{
DECLEAR_DYNCRT_CLASS(Derived, Base)
public:
virtual void Print()
{
cout << "This is derived!" << endl;
}
};
IMPLEMENT_DYNCRT_CLASS(Derived)
int main()
{
Base* base = Base::create("Base");
if (base)
base->Print();
system("pause");
return 0;
}
參考地址:http://blog.csdn.net/freefalcon/article/details/109275