做者:掌門狗html
時間:2008-12-01 20:10ios
分類:C_plusplusthis
【目的】指針
根據字符串運行時刻動態產生類對象htm
【編譯運行環境】對象
Windows/ VC6
Linux / g++blog
【使用方法】字符串
1. 首先須要定義類,好比 B 從 A 派生出來get
2. 調用如下語句註冊B類:
DYN_DECLARE(B);
3. 此後只需用類名對應的字符串就能夠建立該類的對象:
A * p1 = (A *) DYN_CREATE("B");
此時建立的是 B 的對象(downcast)
4. 若是某個類未註冊(經過DYN_DECLARE),DYN_CREATE 返回NULL指針
【實現】
/***************************************************************************/
/* dynclass.h */
/***************************************************************************/
#ifndef __DYNAMIC_H__
#define __DYNAMIC_H__
#include < cstdio >
#include < string >
#include < typeinfo >
#if !defined ( DYN_DECLARE )
#define DYN_DECLARE(class_name) DYN_CLASS::CFactory class_name
#endif
#if !defined ( DYN_CREATE )
#define DYN_CREATE(class_name) DYN_CLASS::Create(class_name)
#endif
namespace DYN_CLASS
{
/* create object by class name */
void * Create( const char * class_name );
/* interface of class factory*/
class CAbstractFactory
{
public:
virtual void * Create( const char * class_name ) = 0;
};
/* list of class factory */
class CFactoryList
{
friend void * Create( const char * class_name );
static CFactoryList * _head;
CFactoryList * m_next;
CAbstractFactory * m_item;
public:
CFactoryList( CAbstractFactory * fact );
virtual ~CFactoryList( void );
};
/* ctor of CFactoryList, add a class factory to list */
inline CFactoryList::CFactoryList( CAbstractFactory * fact )
: m_item( fact )
{
m_next = _head;
_head = this;
}
#if defined ( _MSC_VER )
/* disable warning for the following line : CFactory( void ): m_item( this ) {} */
#pragma warning(disable : 4355)
#endif
/* realization of class factory */
template
class CFactory: public CAbstractFactory
{
static t _object;
CFactoryList m_item;
public:
/* add itself to list of class factory when constructed */
CFactory( void ) : m_item( this ) {}
virtual ~CFactory() {}
/* create object of this class if matched */
void * Create( const char * class_name )
{
std::string strClassName;
#if defined (WIN32 )
strClassName = ( "class " );
#else
char szSize[4] = {0};
sprintf(szSize, "%d", strlen(class_name) );
strClassName = szSize;
#endif
strClassName += class_name;
/* RTTI support */
return !strcmp( typeid(_object).name(), strClassName.c_str() )
? (void*)( new t ) : 0 ;
}
};
}
#endif /* __DYNAMIC_H__ */
/***************************************************************************/
/* dynclass.cpp */
/***************************************************************************/
#include " dynclass.h "
namespace DYN_CLASS
{
CFactoryList * CFactoryList::_head = 0;
void *Create( const char * class_name )
{
void * new_object = 0;
const CFactoryList * cur = CFactoryList::_head;
for( ; cur ; cur = cur->m_next )
{
/* if class_name matched, object will then be created and returned */
if( new_object = cur->m_item->Create(class_name) )
{
break;
}
}
return new_object;
}
/* delete linkage from CFactoryList when some class factory destroyed */
CFactoryList::~CFactoryList( void )
{
CFactoryList ** m_nextp = &CFactoryList::_head;
for( ; *m_nextp ; m_nextp = &(*m_nextp)->m_next )
if( *m_nextp == this )
{
*m_nextp = (*m_nextp)->m_next;
break;
}
}
}
/***************************************************************************/
/* main.cpp */
/***************************************************************************/
#include < iostream >
#include " dynclass.h "
class CBase
{
public:
CBase(){ std::cout << "CBase ctor" << std::endl; }
virtual ~CBase(){ std::cout << "CBase dtor" << std::endl; }
};
class CDerived1 : public CBase
{
public:
CDerived1(){ std::cout << "CDerived1 ctor" << std::endl; }
virtual ~CDerived1(){ std::cout << "CDerived1 dtor" << std::endl; }
};
class CDerived2 : public CBase
{
public:
CDerived2(){ std::cout << "CDerived2 ctor" << std::endl; }
virtual ~CDerived2(){ std::cout << "CDerived2 dtor" << std::endl; }
};
int main(void)
{
DYN_DECLARE(CDerived1);
DYN_DECLARE(CDerived2);
CBase * p1 = (CBase *) DYN_CREATE("CDerived1");
CBase * p2 = (CBase *) DYN_CREATE("CDerived2");
std::cout << std::endl;
if ( p1 != NULL )
{
delete p1;
}
if ( p2 != NULL )
{
delete p2;
}
return 0;
}
/* result:
CBase ctor
CDerived1 ctor
CBase ctor
CDerived2 ctor
CDerived1 dtorCBase dtorCDerived2 dtorCBase dtor*/