C++ 动态生成对象
1、啰嗦一下
说起C++,很多人都觉着难学,其实我也是这么觉着的,在这个移动端火到爆的时代,我都想改行了,移动端做东西那都是现有的第三方库,拿来就可以用,而且稳定性好,开发速度快,而且最关键的是出东西。
在谈一谈动态生成对象,为什么强大的C++不支持呢?想用这样功能的人都必须自己实现一套这样的逻辑。
2、实现理由
有时候开发真是有些矛盾,例如:1、实现一个功能可以使用大量相似的代码、也可以使用模板,那我们怎么选择呢? 2、如果实现一个类之后,他有大量的属性,而且这些属性都需要set和get方法,那么我们还是要Ctrl +C和Ctrl+V吗?如果有好多这样的类,还是Ctrl+C和Ctrl+V吗?对于第一个问题,一个力求上进开发人员,我相信他会选择模板,第二个问题的答案,也就是我们这篇文章所需要讲到的东西,动态生成对象、序列化和反序列化。
3、实现思路
其实这个功能实现起来代码量还是比较少的,就是使用大量的宏和工厂模式
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);
- #define DESTORY_CLASS(className)\
- if (className##Namedc){delete className##Namedc; className##Namedc = nullptr;}
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:
- };
4、测试类
- 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 = ;
- int m_Sex = ;
- };
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 = ;
- int pSex = ;
- 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测试结果
5、序列化和反序列化
本片文章主要讲解的是动态生成对象,并没有打算深入的去剖析系列化和反序列化的模块,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();
- }
- }
- }
- }
说到对象序列化,我就觉得有一个问题比较难搞定,对象包含对象,也就是递归序列化,如果涉及到判断递归那么我们可能还需要自己实现一套结构,用于表示当前对象是否包含其他对象,是否需要继续递归序列化的问题。后面有机会我会对此问题在专门做一篇文章加以解释。
6、demo下载地址
![]() |
![]() |
很重要--转载声明
- 本站文章无特别说明,皆为原创,版权所有,转载时请用链接的方式,给出原文出处。同时写上原作者:朝十晚八 or Twowords
- 如要转载,请原文转载,如在转载时修改本文,请事先告知,谢绝在转载时通过修改本文达到有利于转载者的目的。
C++ 动态生成对象的更多相关文章
- python-根据字符串动态生成对象eval
# -*- coding: utf-8 -*- stock1={ 'stockName':"沈阳机床", ", 'averagePrice_yesterday':34.0 ...
- cglib根据数据动态生成对象
最近有个任务:根据查询SQL直接导出报表 实现关键是,怎么根据sql查询的数据动态生成对象列表,想到Cglib动态代理实现 废话少说,上代码: 定义动态生成Java Bean类: import jav ...
- 利用runtime动态生成对象?
利用runtime我们能够动态生成对象.属性.方法这特性 假定我们要动态生成DYViewController,并为它创建属性propertyName 1)对象名 NSString *class = @ ...
- WPF 动态生成对象属性 (dynamic)
原文:WPF 动态生成对象属性 (dynamic) 项目中列行的数据 都需要动态生成 所以考虑到对象绑定 可需要一个动态生成属性的意思 缺点 加载速度会慢 很明显的慢 解决办法 尽量减轻动态属性的量 ...
- 根据xml配置使用反射动态生成对象
web.xml <?xml version="1.0" encoding="UTF-8"?> <web-app xmlns="htt ...
- WPF Datagrid 动态生成列 并绑定数据
原文:WPF Datagrid 动态生成列 并绑定数据 说的是这里 因为列头是动态加载的 (后台for循环 一会能看到代码) 数据来源于左侧列 左侧列数据源 当然num1 属于临时的dome使用 可 ...
- log4j2 不使用配置文件,动态生成logger对象
大家平时使用Log4j一般都是在classpath下放置一个log4j的配置文件,比如log4j.xml,里面配置好Appenders和Loggers,但是前一阵想做某需求的时候,想要的效果是每一个任 ...
- 利用StringList对象来管理这些动态生成的对象
如果程序需要动态创建大量的对象,那么我们可以利用StringList对象来管理这些动态生成的对象.1.创建StringList对象:OBJ := TStringList.Create; 2.保存动态生 ...
- JS 动态生成JSON对象
JS 动态生成JSON对象,一般用到JSON传递参数的时候,会用到. function onGeneratedRow(columnsResult) { var jsonData = {}; colum ...
随机推荐
- Java - 数组排序 -- 浅析稳定性与复杂度
上次我们了解了对数组的基本操作,那么谈到数组,我们就不得不谈谈数组的排序 什么是排序 排序是计算机内经常进行的一种操作,其目的是将一组“无序”的记录序列调整为“有序”的记录序列 -- 百度百科 排序是 ...
- 2018-2019-2 网络对抗技术 20165319 Exp2 后门原理与实践
后门的基本概念及基础问题 后门程序就是留在计算机系统中,供某位特殊使用者通过某种特殊方式控制计算机系统的途径. [1] 后门程序,跟我们通常所说的"木马"有联系也有区别.联系在于: ...
- QEMU, a Fast and Portable Dynamic Translator-Fabrice Bellard-翻译
Abstract We present the internals of QEMU, a fast machine emulator using an original portable dynami ...
- 用python写一个非常简单的QQ轰炸机
闲的没事,就想写一个QQ轰炸机,按照我最初的想法,这程序要根据我输入的QQ号进行轰炸,网上搜了一下,发现网上的案列略复杂,就想着自己写一个算了.. 思路:所谓轰炸机,就是给某个人发很多信息,一直刷屏, ...
- 思科与华为RIP配置区别
华为配置图如下: 思科配置图如下: 配置原理一样,除了配置命令有点区别:华为进入RIP的命令为:rip 1 思科进入RIP的命令为:router rip
- 最简单的原生js和jquery插件封装
最近在开发过程中用别人的插件有问题,所以研究了一下,怎么封装自己的插件. 如果是制作jquery插件的话.就将下面的extend方法换成 $.extend 方法,其他都一样. 总结一下实现原理: 将 ...
- IDEA+Tomcat+Maven+SpringMVC基于Java注解配置web工程
1.在IDEA中新建Maven工程,使用archetype. 2.添加Maven依赖 <dependencies> <dependency> <groupId>ju ...
- SpringBoot报错:Invalid bound statement (not found)
错误原因: 没有发现Mybatis配置文件的路径 解决方法: 检查Mapper包名与xml文件标签的namespace数据名称是否相同 <mapper namespace="com.t ...
- synchronized 与 volatile 原理 —— 内存屏障的重要实践
单例模式的双重校验锁的实现: 第一种: private static Singleton _instance; public static synchronized Singleton getInst ...
- 微信小程序学习笔记(三)
一般setData方法多用于点击后改变页面信息或者刷新后与后台交互获取最新的信息 注意: 直接修改 this.data 而不调用 this.setData 是无法改变页面的状态的,还会造成数据不一致 ...