91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

C++動態的生成對象的方法

發布時間:2021-02-20 13:39:22 來源:億速云 閱讀:219 作者:小新 欄目:編程語言

這篇文章主要介紹了C++動態的生成對象的方法,具有一定借鑒價值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。

實現理由

有時候開發真是有些矛盾,例如:1、實現一個功能可以使用大量相似的代碼、也可以使用模板,那我們怎么選擇呢? 2、如果實現一個類之后,他有大量的屬性,而且這些屬性都需要set和get方法,那么我們還是要Ctrl +C和Ctrl+V嗎?如果有好多這樣的類,還是Ctrl+C和Ctrl+V嗎?對于第一個問題,一個力求上進開發人員,我相信他會選擇模板,第二個問題的答案,也就是我們這篇文章所需要講到的東西,動態生成對象、序列化和反序列化。

實現思路

其實這個功能實現起來代碼量還是比較少的,就是使用大量的宏和工廠模式

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、然后在寫一個方便類,這個類僅僅是為了注冊方便,當這個類被聲明的時候,即注冊一個類到工廠中

class CDynamicClass
{
public:
 CDynamicClass(const std::string & name, const CreateClass & method)
 {
 CClassFactory::IntanceFactory().RegistClass(name, method);
 }
};

3、2個關鍵的宏,這兩個宏一個是用于CDynamicClass靜態對象的,一個是用于初始化CDynamicClass對象的,作用請看上一小節,呵呵呵,其實就是注冊宏的參數類到工廠

 #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宏用于注冊屬性的相關接口,ACCESS_REGISTER宏是把屬性名字和對象的屬性調用接口記錄起來,方便以后設置屬性

#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類是一個測試類,用于測試第三節所寫的動態生成對象是否正確,RegistProperty接口里邊是對屬性的注冊

1、測試main函數

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、效果結果截圖

C++動態的生成對象的方法

圖1 CHelloClass測試結果

序列化和反序列化

本片文章主要講解的是動態生成對象,并沒有打算深入的去剖析系列化和反序列化的模塊,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();
   }
  }
 }
}

說到對象序列化,我就覺得有一個問題比較難搞定,對象包含對象,也就是遞歸序列化,如果涉及到判斷遞歸那么我們可能還需要自己實現一套結構,用于表示當前對象是否包含其他對象,是否需要繼續遞歸序列化的問題。后面有機會我會對此問題在專門做一篇文章加以解釋。

感謝你能夠認真閱讀完這篇文章,希望小編分享的“C++動態的生成對象的方法”這篇文章對大家有幫助,同時也希望大家多多支持億速云,關注億速云行業資訊頻道,更多相關知識等著你來學習!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

大冶市| 张家川| 聂拉木县| 高清| 尤溪县| 新津县| 通海县| 博爱县| 浙江省| 庆云县| 青河县| 望谟县| 招远市| 新源县| 汝南县| 镇安县| 瓦房店市| 江北区| 蕉岭县| 米脂县| 奈曼旗| 郑州市| 彭山县| 宣化县| 临海市| 清丰县| 桐乡市| 东至县| 上高县| 夏邑县| 汝阳县| 武汉市| 砀山县| 富阳市| 礼泉县| 万州区| 英山县| 城市| 蓝田县| 聂荣县| 铁岭县|