C++如何動態(tài)的生成對象詳解
前言
可能說起C++大多數(shù)人都覺著難學(xué),其實我也是這么覺著的,在這個移動端火到爆的時代,我都想改行了,移動端做東西那都是現(xiàn)有的第三方庫,拿來就可以用,而且穩(wěn)定性好,開發(fā)速度快,而且最關(guān)鍵的是出東西。再談一談動態(tài)生成對象,為什么強大的C++不支持呢?想用這樣功能的人都必須自己實現(xiàn)一套這樣的邏輯。
實現(xiàn)理由
有時候開發(fā)真是有些矛盾,例如:1、實現(xiàn)一個功能可以使用大量相似的代碼、也可以使用模板,那我們怎么選擇呢? 2、如果實現(xiàn)一個類之后,他有大量的屬性,而且這些屬性都需要set和get方法,那么我們還是要Ctrl +C和Ctrl+V嗎?如果有好多這樣的類,還是Ctrl+C和Ctrl+V嗎?對于第一個問題,一個力求上進開發(fā)人員,我相信他會選擇模板,第二個問題的答案,也就是我們這篇文章所需要講到的東西,動態(tài)生成對象、序列化和反序列化。
實現(xiàn)思路
其實這個功能實現(xiàn)起來代碼量還是比較少的,就是使用大量的宏和工廠模式
1、寫一個工廠類,專門用于生成對象
typedef void * (* CreateClass)(void);
class CClassFactory
{
public:
static CClassFactory & IntanceFactory();
public:
void * CreateObject(const std::string & className);
void RegistClass(const std::string & name, const CreateClass & method);
private:
std::map<std::string, CreateClass> m_classMap;
};
2、然后在寫一個方便類,這個類僅僅是為了注冊方便,當(dāng)這個類被聲明的時候,即注冊一個類到工廠中
class CDynamicClass
{
public:
CDynamicClass(const std::string & name, const CreateClass & method)
{
CClassFactory::IntanceFactory().RegistClass(name, method);
}
};
3、2個關(guān)鍵的宏,這兩個宏一個是用于CDynamicClass靜態(tài)對象的,一個是用于初始化CDynamicClass對象的,作用請看上一小節(jié),呵呵呵,其實就是注冊宏的參數(shù)類到工廠
#define DECLARE_CLASS(className)\ std::string className##Name;\ static CDynamicClass * className##Namedc; #define IMPLEMENT_CLASS(className)\ CDynamicClass * className::className##Namedc = new CDynamicClass(#className, className::Instance);
4、2個屬性宏,ACCESS_INTERFACE宏用于注冊屬性的相關(guān)接口,ACCESS_REGISTER宏是把屬性名字和對象的屬性調(diào)用接口記錄起來,方便以后設(shè)置屬性
#define ACCESS_INTERFACE(classType, type, name, describe)\
public:\
std::string m_Describe##name = #describe;\
inline static void Set##name(CBaseClass * cp, void * value){\
classType * tp = (classType *)cp;\
tp->m_##name = *(type *)value;\
}\
inline type Get##name(void) const {\
return m_##name;\
}\
inline std::string Get##name##Describe(){ \
return m_Describe##name;\
}
#define ACCESS_REGISTER(name)\
m_propertyMap.insert({ #name, Set##name });
5、基類,所有對象的基類,m_propertyMap成員是存儲屬性和屬性對于的set接口對
class CBaseClass
{
public:
CBaseClass() {}
virtual ~CBaseClass() {}
public:
std::map<std::string, SetValueProperty> m_propertyMap;
private:
};
測試類
class CHelloClass : public CBaseClass
{
public:
DECLARE_CLASS(CHelloClass);
ACCESS_INTERFACE(CHelloClass, int, Age, "年齡")
ACCESS_INTERFACE(CHelloClass, int, Sex, "性別")
public:
CHelloClass();
virtual ~CHelloClass();
public:
static void * Instance();
public:
virtual void RegistProperty( );
protected:
int m_Age = 0;
int m_Sex = 0;
};
CHelloClass類是一個測試類,用于測試第三節(jié)所寫的動態(tài)生成對象是否正確,RegistProperty接口里邊是對屬性的注冊
1、測試main函數(shù)
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
CHelloClass * pVar = (CHelloClass*)CClassFactory::IntanceFactory().CreateObject("CHelloClass");
if (pVar)
{
int pAge = 2;
int pSex = 1;
pVar->m_propertyMap["Age"](pVar, &pAge);
pVar->m_propertyMap["Sex"](pVar, &pSex);
std::cout << pVar->GetAgeDescribe() << pVar->GetAge() << std::endl;
std::cout << pVar->GetSexDescribe() << pVar->GetSex() << std::endl;
}
return a.exec();
}
2、效果結(jié)果截圖
圖1 CHelloClass測試結(jié)果
序列化和反序列化
本片文章主要講解的是動態(tài)生成對象,并沒有打算深入的去剖析系列化和反序列化的模塊,demo中也有一小部分的序列化代碼,主要是使用tinyxml2來讀文件,代碼如下:
void DynamicObject::Deserialize()
{
tinyxml2::XMLDocument doc;
if (tinyxml2::XML_NO_ERROR == doc.LoadFile("D:\\example\\paint\\DynamicCreateObject\\test.xml"))
{
if (tinyxml2::XMLNode * rootNode = doc.FirstChildElement("Ojbectlist"))
{
const char * rootText = rootNode->ToElement()->Attribute("name");
tinyxml2::XMLElement * element = rootNode->FirstChildElement("Object");
while (element)
{
const char * objectName = element->Attribute("name");
tinyxml2::XMLElement * propertyElement = element->FirstChildElement("Property");
while (propertyElement)
{
const char * propertyName = propertyElement->Attribute("name");
const char * propertyValue = propertyElement->Attribute("value");
}
tinyxml2::XMLNode * nextNode = element->NextSibling();
if (nextNode == nullptr)
{
break;
}
element = nextNode->ToElement();
}
}
}
}
說到對象序列化,我就覺得有一個問題比較難搞定,對象包含對象,也就是遞歸序列化,如果涉及到判斷遞歸那么我們可能還需要自己實現(xiàn)一套結(jié)構(gòu),用于表示當(dāng)前對象是否包含其他對象,是否需要繼續(xù)遞歸序列化的問題。后面有機會我會對此問題在專門做一篇文章加以解釋。
demo下載地址
C++動態(tài)生成對象
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,如果有疑問大家可以留言交流。
上一篇:淺談C語言共用體和與結(jié)構(gòu)體的區(qū)別
欄 目:C語言
下一篇:C++11中l(wèi)ambda、std::function和std:bind詳解
本文標(biāo)題:C++如何動態(tài)的生成對象詳解
本文地址:http://www.jygsgssxh.com/a1/Cyuyan/1775.html
您可能感興趣的文章
- 04-02c語言沒有round函數(shù) round c語言
- 01-10如何判斷一個數(shù)是否為2的冪次方?若是,并判斷出來是多少次方
- 01-10深入理解C++中常見的關(guān)鍵字含義
- 01-10使用C++實現(xiàn)全排列算法的方法詳解
- 01-10如何判斷一個數(shù)是否為4的冪次方?若是,并判斷出來是多少次方
- 01-10如何查看進程實際的內(nèi)存占用情況詳解
- 01-10c++中inline的用法分析
- 01-10如何尋找數(shù)組中的第二大數(shù)
- 01-10用C++實現(xiàn)DBSCAN聚類算法
- 01-10全排列算法的非遞歸實現(xiàn)與遞歸實現(xiàn)的方法(C++)


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機閱讀
- 01-10C#中split用法實例總結(jié)
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-10使用C語言求解撲克牌的順子及n個骰子
- 04-02jquery與jsp,用jquery
- 01-10delphi制作wav文件的方法
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 01-11ajax實現(xiàn)頁面的局部加載
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 08-05織夢dedecms什么時候用欄目交叉功能?


