use*_*547 2 c++ templates boost-mpl
我想用简单的C++语法创建一个简单的工厂方法:
void *createObject(const char *str,...)
{
if(!strcmp("X",str))
return new X(...);
}
Run Code Online (Sandbox Code Playgroud)
我无法弄清楚这个的语法.我一直在看模板元编程并使用mpl :: vectors,但我不知道如何传递这种语法.我想尽可能避免使用C va_lists,并采用如上所述的干净语法.
这对C++ 11来说是一个更好的方法:
template< typename ...Args >
std::shared_ptr<void> createObject( std::string const& name, Args&& ...args )
{
if( name == "X" )
{
return try_make_shared< X >( std::forward< Args >( args )... );
}
/* other cases here*/
return nullptr;
}
template< typename T, typename ...Args >
typename std::enable_if<
std::is_constructible< T, Args >::value
, std::shared_ptr< T >
>::type try_make_shared( Args&&... args )
{
return std::make_shared< X >( std::forward< Args >( args )... );
}
template< typename T, typename ...Args >
typename std::enable_if<
!std::is_constructible< T, Args >::value
, std::shared_ptr< T >
>::type try_make_shared( Args&&... args )
{
throw std::invalid_argument( "The type is not constructible from the supplied arguments" );
return nullptr;
}
Run Code Online (Sandbox Code Playgroud)
与您的代码的差异是
它使用可变参数模板函数而不是省略号参数,因此参数的数量和类型在编译时仍然可用(您不会丢失类型检查).此外,您可以使用非POD类型调用此函数.
它返回一个shared_ptr<void>而不是一个普通的void*.这允许您在工厂内控制一旦所有对它的引用都消失后应该如何清理对象.用户无需知道或关心他是否应该调用标准delete,或者可能是deleteObject您工厂的方法.
更新:对于那些建议unique_ptr,你可以在这里阅读有关shared_ptr带来的可能性.只能返回指向已new分配对象的指针的受限工厂可以并且应该使用a unique_ptr.