DECLARE_DYNAMIC

摘要:
DECLARE_DYNAMICDECLARE_DYNCREATE包含了DECLARE_DYNAMIC的功能,并且可以在运行过程中动态创建对象。如果需要动态创建类对象,需要使用这个宏定义。IMPLEMENT_DYNAMIC是实现“运行时类型识别”宏,与之相对应的是DECLARE_DYNAMIC。也就是说你在.CPP文件中如果看见有IMPLEMENT_DYNAMIC,则在.H文件中必定有DECLARE_DYNAMIC的声明。

DECLARE_DYNAMIC(class_name)

DECLARE_DYNCREATE包含了DECLARE_DYNAMIC的功能,并且可以在运行过程中动态创建对象。如果需要动态创建类对象,需要使用这个宏定义。

IMPLEMENT_DYNAMIC是实现“运行时类型识别”宏,与之相对应的是DECLARE_DYNAMIC(声明“运行时类型识别”宏)。也就是说你在.CPP文件中如果看见有IMPLEMENT_DYNAMIC,则在.H文件中必定有DECLARE_DYNAMIC的声明。
DECLARE_DYNAMIC/DEClARE_DYNAMIC是为了确定运行时对象属于哪一个类而定义的宏。
DEClARE_DYNCREATE/IMPLEMENT_DYNCREATE是为了“动态创建"类的实例而定义的宏。new可以用来创建对象,但不是动态的。比如说,你要在程序中实现根据拥护输入的类名来创建类的实例,下面的做法是通不过的:
char szClassName[60];
cin >> szClassName;
CObject* pOb=new szClassName; //通不过
这里就要用到DEClARE_DYNCREATE/IMPLEMENT_DYNCREATE定义的功能了。

定义:

//////////////////////////////////////////////////////////////////////////////
// Helper macros for declaring CRuntimeClass compatible classes
#ifdef _AFXDLL
#define DECLARE_DYNAMIC(class_name) /
protected: /
static CRuntimeClass* PASCAL _GetBaseClass(); /
public: /
static const CRuntimeClass class##class_name; /
static CRuntimeClass* PASCAL GetThisClass(); /
virtual CRuntimeClass* GetRuntimeClass() const; /
#define _DECLARE_DYNAMIC(class_name) /
protected: /
static CRuntimeClass* PASCAL _GetBaseClass(); /
public: /
static CRuntimeClass class##class_name; /
static CRuntimeClass* PASCAL GetThisClass(); /
virtual CRuntimeClass* GetRuntimeClass() const; /
#else
#define DECLARE_DYNAMIC(class_name) /
public: /
static const CRuntimeClass class##class_name; /
virtual CRuntimeClass* GetRuntimeClass() const; /
#define _DECLARE_DYNAMIC(class_name) /
public: /
static CRuntimeClass class##class_name; /
virtual CRuntimeClass* GetRuntimeClass() const; /
#endif

引用:

DECLARE_DYNAMIC/IMPLEMENT_DYNAMIC
#define DECLARE_DYNAMIC(class_name)/
public:/
static CRuntimeClass class##class_name;/
//声明一个类型为CRuntimeClass的静态public成员变量,变量名是由字符串"class"
//与所指定的类的类名组成。举例而言,如果你写DECLARE_DYNAMIC(CMyView),则等于声明了一个
// static CRuntimeClass classCMyView静态变量
virtual CRuntimeClass* GetRuntimeClass() const;/
//声明一个虚函数,函数名为GetRuntimeClass,返回值为CRuntimeClass类型的指针
//无参数,并且是个const函数
#define IMPLEMENT_DYNAMIC(class_name,bass_class_name)/
_IMPLEMENT_RUNTIMECLASS(class_name,base_class_name,0xFFFF,NULL)
#define _IMPLEMENT_RUNTIMECLASS(class_name,base_class_name,wSchema,pfnNew)/
static char _lpsz##class_name[]= #class_name;/
//定义一个C类型字符串静态变量,变量名由"_lpsz"和指定类的类名组成,变量值为该指定类型的名字
//比如是CMyView,那么定义的就是static char _lpszCMyView="CMyView";
CRuntimeClass class_name::class##class_name = {/
_lpsz##class_name,sizeof(class_name),wSchema,pfnNew,/
RUNTIME_CLASS(base_class_name),NULL};/
//给之前在DECLARE_DYNAMIC里定义的CRuntimeClass类型的静态成员变量赋值
//当然,除最后一个m_pNextClass没有赋值(赋值为NULL,它由下面的结构处理)
static AFX_CLASSINIT _init_##class_name(&class_name::class##class_name);/
//初始化一个名为"_init_##class_name"的AFX_CLASSINIT静态结构,主要作用是给指定的class_name的
//class##class_name静态变量的最后一个成员m_pNextClass赋值,具体见下面解释AFX_CLASSINIT中
CRuntimeClass* class_name::GetRuntimeClass() const/
{ return &class_name::class##class_name;}/
//之前在DECLARE_DYNAMIC里定义的GetRuntimeClass的实现,很简单,就一个return语句。
#define RUNTIME_CLASS(class_name)/
(&class_name::class##class_name)
//这部分之所以单独define出一个宏,主要是为了方便从某个指定的class直接得到它的CRuntimeclass静态成员
//以下是解释AFX_CLASSINIT结构,注意,这不是一个宏
//为了看得更加清楚,我按照struct定义的惯常格式来写这个struct的定义
struct AFX_CLASSINIT {
AFX_CLASSINIT(CRuntimeClass *pNewClass);
};
AFX_CLASSINIT::AFX_CLASSINIT(CRuntimeClass *pNewClass)
{
pNewClass->m_pNextClass = CRuntimeClass::pFirstClass;
//让m_pNextClass指向pFirstClass所指的CRuntimeClass变量
CRuntimeClass::pFirstClass = pNewClass;
//让pFirstClass指向pNewClass所指的变量,也就是本class的CRuntimeClass静态变量
}

免责声明:文章转载自《DECLARE_DYNAMIC》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇idea 配置mapper.xml代码提示Unity 角色控制器 Charactor Controller下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

Math工具类的使用

1.包:java.lang 不需要导包 2.Math 类包含用于执行基本数学运算的方法,如初等指数、对数、平方根和三角函数。 特点: 该类中的方法都是静态方法,所以可以直接使用类名.方法名(实参)调用 3.常用成员方法: public static double abs(double a) :返回 double 值的绝对值。 public static d...

静态代码块

static {//静态代码块 }   关于静态代码块,要注意的是: 它是随着类的加载而执行,只执行一次,并优先于主函数。具体说,静态代码块是由类调用的。类调用时,先执行静态代码块,然后才执行主函数的。 静态代码块其实就是给类初始化的,而构造代码块是给对象初始化的。 静态代码块中的变量是局部变量,与普通函数中的局部变量性质没有区别。 一个类中可...

String的intern方法的用处

今天第一次翻看Effective java,在其第一个item中讲静态工厂方法的有点的时候说到“它们每次被调用 的时候,不要非得创建一个新的对象”并在结尾处提到---"String.intern方法以一种有限的形式实现了这 种优化",由于以前没用过intern这个方法,所以去查了查,并把自己的理解写下来供以后参考首先来看其中文API的描述: intern...

Java系统属性与Preferences API的简单介绍

系统属性在和Preferences API都是键值对,前者只能当前应用程序中共享数据,而后者可以在用户的各个应用或用户之间共享数据。 系统属性 Java 的系统属性决定了 Java 程序实际运行的环境,默认情况下,JVM 启动时采用系统默认属性值。系统属性顾名思义是指与用户程序相关的操作系统配置信息以及软件信息,位于java.lang包中。 设计到的方法:...

JAVA-初步认识-第七章-static关键字-数据共享

一. 介绍一个新的关键字,并不陌生,static在写主函数的时候,经常使用到。 static是一个关键字,同时也是一个修饰符。它到底有什么用呢?到底什么时候用呢? 下面的截图中,刚开始对于描述类的书写是String name。并没有加上修饰符private,是为了方便调用,同时也没有书写,get,set两个固定的方法。(对于描述类的书写还是欠缺很多,前面...

java 单例模式

单例模式 : 一个类创建的 任何对象都是同一个只存在一个实例,都只开辟同一块内存空间 单例模式就是不管外部如何创建 都只是创建一个对象 对象操作的也是只个唯一对象 饿汉式: public classSingleton { public static voidmain(String[] args) { Pattern patternO...