1.應(yīng)用:
也許需要一個腳本,腳本中有一個參數(shù)傳遞類名,你需要通過類名來動態(tài)的創(chuàng)建這個對象,這時候就需要動態(tài)創(chuàng)建了
2.原理:
C++不是動態(tài)語言,所以沒法從語言機制上實現(xiàn)類的動態(tài)創(chuàng)建,但這樣的需求卻有可能存在,一個類似的例子便是MFC中CWnd類的Create方法,其第一個參數(shù)為Window Class的名字,這就允許用戶通過class的名字來創(chuàng)建相應(yīng)的窗口。
要想實現(xiàn)這一點,
必須有一個“管理中心”,用于登記類的名字,并且通過名字能夠調(diào)用對應(yīng)的創(chuàng)建方法來創(chuàng)建相應(yīng)的實例。結(jié)合類工廠的設(shè)計思想,這里我們讓一套繼承體系中的基類作為“管理中心”,由它來維護所有派生類的必要信息,包括類名和工廠函數(shù),這二者必須建立起映射關(guān)系,map是不錯的選擇。定義了一個派生類后,它就自動向基類進行注冊
3.實現(xiàn):
可以用宏來代替:
/**
實現(xiàn)原理:
基類里面用map管理所有需要動態(tài)創(chuàng)建 【類名-創(chuàng)建對象的函數(shù)指針】
基類聲明中需要添加 DECLEAR_DYNCRT_BASE 宏,cpp實現(xiàn)中要添加 IMPLEMENT_DYNCRT_BASE 宏
派生類聲明中需要添加 DECLEAR_DYNCRT_CLASS 宏,cpp實現(xiàn)中要添加 IMPLEMENT_DYNCRT_CLASS 宏
派生類宏中創(chuàng)建了一個全局對象向基類map中注冊【類名-創(chuàng)建對象的函數(shù)指針】
*/
#include <map>
// 聲明具有動態(tài)創(chuàng)建的基類
#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;
// 用于實現(xiàn)基類
#define IMPLEMENT_DYNCRT_BASE(base) \
std::map<const char*, base::ClassGen> base::class_set;
// 用于聲明一個能夠被動態(tài)創(chuàng)建的類(用一個全局對象進行注冊)
#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; \
}
// 用于實現(xiàn)一個能被動態(tài)創(chuàng)建的類
#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;
}
參考地址:
http://blog.csdn.net/freefalcon/article/details/109275
posted on 2012-08-30 12:15
風輕云淡 閱讀(2133)
評論(0) 編輯 收藏 引用 所屬分類:
C++