我正在使用带有CRTP的模板类来实现克隆模式,并使用第二个模板参数Base来实现多个继承级别。当我尝试调用间接基类的构造函数时,出现编译器错误。
class B
{
public:
B() {} //trivial constructor
virtual B* clone()=0;
};
template<class Base, class Derived>
class Clonable
:public Base //weird, I know
{
public:
virtual B* clone() {return new Derived(*this);}
};
class D1 : public Clonable<B, D1>
{
public:
D1(int a); //non-trivial constructor. Different signature than B
};
class D2 : public Clonable<D1, D2>
{
public:
D2(int a): D1(a) {} //compiler error here
}
到目前为止,我遇到的唯一解决方案是在Cloneable中使用可变参数模板构造函数,但是我的编译器(VC ++ 11)尚未实现它们。
最佳答案
您需要让克隆的“中间人”类转发构造函数参数,或者更好(Luc Danton建议这样做)使用C ++ 11构造函数继承。
因此,在C ++ 11中很容易做到这一点,但在C ++ 03或当前尚不支持C ++ 11参数转发或构造函数继承的编译器(如Visual C ++)中,要实现此目的并不容易10。
在我的旧博客文章"3 ways to mix in a generic cloning implementation"中,讨论了在C ++ 03中使用辅助参数转发器类实现此目的的一种方法。然后,中间人(克隆实现)类如下所示:
template< class Derived, class Base >
class WithCloningOf
: public progrock::cppx::ConstructorArgForwarder< Base >
{
protected:
virtual WithCloningOf* virtualClone() const
{
return new Derived( *static_cast< Derived const* >( this ) );
}
public:
template< class ArgPack >
WithCloningOf( ArgPack const& args )
: progrock::cppx::ConstructorArgForwarder< Base >( args )
{}
std::auto_ptr< Derived > clone() const
{
return std::auto_ptr< Derived >(
static_cast< Derived* >( virtualClone() )
);
}
};
我在较早的博客文章中讨论了与C ++ 03兼容的
ConstructorArgForwarder
。它看起来像这样:template< typename Type >
class ConstructorArgForwarder
: public Type
{
public:
typedef Type Base;
// TODO: remove
virtual ~ConstructorArgForwarder() {}
ConstructorArgForwarder( EmptyArgPack const& )
: Base()
{}
template< class T01 >
ConstructorArgForwarder(
ArgPack< T01 > const& args
)
: Base( args.a01 )
{}
template< class T01, class T02 >
ConstructorArgForwarder(
ArgPack< T01, T02 > const& args
)
: Base( args.a01, args.a02 )
{}
template< class T01, class T02, class T03 >
ConstructorArgForwarder(
ArgPack< T01, T02, T03 > const& args
)
: Base( args.a01, args.a02, args.a03 )
{}
// And more, up to max 12 arguments.
};
依次使用参数包类
ArgPack
(好吧,类模板),它看起来像这样:enum NoArg {};
template<
class T01 = NoArg, class T02 = NoArg, class T03 = NoArg,
class T04 = NoArg, class T05 = NoArg, class T06 = NoArg,
class T07 = NoArg, class T08 = NoArg, class T09 = NoArg,
class T10 = NoArg, class T11 = NoArg, class T12 = NoArg
>
struct ArgPack;
template<
>
struct ArgPack<
NoArg, NoArg, NoArg, NoArg, NoArg, NoArg,
NoArg, NoArg, NoArg, NoArg, NoArg, NoArg
>
{};
typedef ArgPack<
NoArg, NoArg, NoArg, NoArg, NoArg, NoArg,
NoArg, NoArg, NoArg, NoArg, NoArg, NoArg
> EmptyArgPack;
inline ArgPack<> args() { return ArgPack<>(); }
template<
class T01
>
struct ArgPack<
T01, NoArg, NoArg, NoArg, NoArg, NoArg,
NoArg, NoArg, NoArg, NoArg, NoArg, NoArg
>
{
T01 const& a01;
ArgPack( T01 const& v01 )
: a01( v01 )
{}
};
template< class T01 >
inline ArgPack< T01 >
args( T01 const& a01 )
{
return ArgPack< T01 >( a01 );
}
免责声明:错误可能只是潜入,例如从我的博客复制代码。但是,在我于2010年5月发布该信息时,它就起作用了。
注意:正如我在以上两个博客文章的最后部分中讨论的关于克隆一样,对于C ++ 03,有三种主要的通用处理方法,其中简单的宏以较高的优势击败了其他两种方法。但是,对于C ++ 11,您在此处选择的“中间人”方法似乎更好。通过优势的“横向继承”只是复杂而低效的,但是如果您限于C ++ 03,那么请考虑使用一个简单的宏!
注意2:上次我建议做实用且明智的事情时,我的想法被否定了(大概是Reddit的孩子们)。但是从那时起,我就不再关心SO代表积分,尤其是downvotes。因此,很高兴地,我现在可以再次提供很好的建议,就像在Usenet的旧时代一样,只是不理会那些沮丧的孩子对某些单词的无意识反应。 :-)
关于c++ - 带CRTP的转发构造函数,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/9520745/