根據字符串動態建立類對象

標籤: 筆記 c++spa

【目的】指針

根據字符串運行時刻動態產生類對象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*/

相關文章
相關標籤/搜索