当前位置: 首页 > 编程笔记 >

C++如何动态的生成对象详解

宰父智敏
2023-03-14
本文向大家介绍C++如何动态的生成对象详解,包括了C++如何动态的生成对象详解的使用技巧和注意事项,需要的朋友参考一下

前言

可能说起C++大多数人都觉着难学,其实我也是这么觉着的,在这个移动端火到爆的时代,我都想改行了,移动端做东西那都是现有的第三方库,拿来就可以用,而且稳定性好,开发速度快,而且最关键的是出东西。再谈一谈动态生成对象,为什么强大的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、效果结果截图

图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();
   }
  }
 }
}

说到对象序列化,我就觉得有一个问题比较难搞定,对象包含对象,也就是递归序列化,如果涉及到判断递归那么我们可能还需要自己实现一套结构,用于表示当前对象是否包含其他对象,是否需要继续递归序列化的问题。后面有机会我会对此问题在专门做一篇文章加以解释。

demo下载地址

C++动态生成对象

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流。

 类似资料:
  • 问题内容: 我知道这听起来是不可能的,但是我的老板告诉我,我必须通过jQuery的AJAX发布调用发送JSON,并且必须具有重复的键。问题是,如果我写这样的话: ,jQuery会将请求发送为 所有这些都是因为Javascript会覆盖具有相同名称的属性。JSON对象是动态生成的,不允许在其中使用数组。有人可以告诉我如何生成具有重复键的JSON对象吗? 非常感谢您的帮助! 问题答案: 据我所知,实际

  • 问题内容: 我正在编写一个包含的程序JButton。每次单击该按钮时,都会将新JTextField的添加到中JPanel。 我的问题是,在用户创建所有JTextFields并用信息填充它们之后,我需要获取每个字段的文本。JTextFields当它们没有实例名称时,如何动态访问它们?有没有一种更好的方法来获取每个文本,而又不知道它们的实例名称。 这是…的actionPerformed事件代码JBut

  • 本文向大家介绍C#动态对象(dynamic)详解(实现方法和属性的动态),包括了C#动态对象(dynamic)详解(实现方法和属性的动态)的使用技巧和注意事项,需要的朋友参考一下 C#的动态对象的属性实现比较简单,如果要实现动态语言那种动态方法就比较困难,因为对于dynamic对象,扩展方法,匿名方法都是不能用直接的,这里还是利用对象和委托来模拟这种动态方法的实现,看起来有点javascript的

  • 我只是想知道类(Oracle提供的rt.jar内部)如java.lang.Object、java.lang.String是如何从.java源文件生成的。我认为,他们的javac不可能编译它们。 我尝试编译包含“类Dummy{}”的Dummy.java [解析已启动RegularFileObject[dummy.java]] [解析完成12毫秒] [源文件的搜索路径:.]

  • 本文向大家介绍js 动态生成json对象、时时更新json对象的方法,包括了js 动态生成json对象、时时更新json对象的方法的使用技巧和注意事项,需要的朋友参考一下 函数不需要 return,因为 json 对象会被函数直接修改。 下面为测试 函数应该有三个参数的,第一个为 json 对象,但应你要求,只写了两个,所以函数只能用于修改 str1。 以上这篇js 动态生成json对象、时时更新

  • 例如,类的ViewBag属性,我们可以动态地获取/设置值,并向该对象添加任意数量的附加字段或属性,这很酷。我想在其他类型的应用程序中使用类似的东西,除了MVC应用程序和类之外。当我试图创建动态对象并设置它的属性时,如下所示: 我有,带有message属性或索引器'<>f__anonymoustype0.a'不能赋值--它在第2行中是只读的。我也认为这不是我想要的。也许有什么课可以让我做一些类似的事