原文链接:
知道有IsKindOf这么个函数,用来检测:(1)对象是否属于指定的类,(2)对象是否属于指定类派生的类,但是一直没用过,今天用了下,提示错误errorC2039: 'classXXX' : is not a member of'XXX',查了很多资料,才把它搞明白,记录如下:
1、对使用IsKindOf这个函数的类要在.h文件夹里定义DECLARE_DYNAMIC或DECLARE_SERIAL宏
class CDlgTest : public CDialog
{
DECLARE_DYNAMIC( CDlgTest )
// Construction
public:
CDlgTest(CWnd* pParent = NULL);
// standardconstructor
...
...
};
2、需要在.cpp文件中加入IMPLEMENT_DYNAMIC或IMPLEMENT_SERIAL宏
[cpp] view plaincopy
IMPLEMENT_DYNAMIC(CDlgTest, CDialog)
CDlgTest::CDlgTest(CWnd* pParent )
{
}
3、使用时,CDlgTest dlg;
随后又研究了一下这几个宏,也算明白了IsKindOf的实现原理,做下简单记录:
1、DECLARE_DYNAMIC宏
[cpp] view plaincopy
#define DECLARE_DYNAMIC(class_name)/
public:/
2、IMPLEMENT_DYNAMIC宏
[cpp] view plaincopy
#define IMPLEMENT_DYNAMIC(class_name,bass_class_name)/
#define_IMPLEMENT_RUNTIMECLASS(class_name,base_class_name,wSchema,pfnNew)/
3、IsKindOf函数的实现
// example forCObject::GetRuntimeClass
CAge a(21);
CRuntimeClass* prt =a.GetRuntimeClass();
ASSERT( strcmp(prt->m_lpszClassName, "CAge" )
==0 );
Return Value:
A pointer to the CRuntimeClassstructure corresponding to this object's class; never NULL.
[cpp] view plaincopy
BOOL
CRuntimeClass::IsDerivedFrom(const CRuntimeClass*pClass) const
{
}
BOOL CObject::IsKindOf(const CRuntimeClass* pClass)const
{
ASSERT(this != NULL);
// it better be in valid memory, at least for CObjectsize
ASSERT(AfxIsValidAddress(this, sizeof(CObject)));
// simple SI case
CRuntimeClass* pClassThis = GetRuntimeClass();
return pClassThis->IsDerivedFrom(pClass);
}
4、单行宏的定义
定义单行宏:主要有以下三种用法.
1) 前加##或后加##,将标记作为一个合法的标识符的一部分.注意,不是字符串.多用于多行的宏定义中.例如:
2) 前加#@,将标记转换为相应的字符,注意:仅对单一标记转换有效
3) 前加#,将标记转换为字符串.
其余网上搜索:
主要检查一下对象否从某一类类派生,但此类必须从CObject派生(或间接),并且使用DECLARE_DYNAMIC/DECLARE_SEARIAL等宏
比如我们函数的参数是个基类的指针类型。那么当我外部传过来一个子类的指针时,在函数内部就可以用这种语句来确认传进来的是这个子类的对象,而不是其它子类的对象
ASSERT(xxx->IsKinfOf(RUNTIME_CLASS(yyy))) 判断xxx所指的类是否为yyy的子类!
指针的类型和指针指向的对象的类型可以是不一样的,这就是指针的灵活性;但是这种灵活性是不安全的,因为可以随意转换,所以前面用一个ASSERT判断一下。
其实现在在C++中也可以使用安全的类型转换:static_cast、dynamic_cast等。

3508

被折叠的 条评论
为什么被折叠?



