假设我有一个类定义模板参数,如std::function
语法。
// Class that accept template arguments like std::function
template< class T >
class abstraction;
template< class TAbstract, class ...TDeriveds >
class abstraction< TAbstract(TDeriveds...) >
{
public:
using abstract_component_t = TAbstract;
abstraction()
{
int i = 0;
}
};
// Abstract base class
class base
{
public:
virtual void func() = 0;
};
// Derived class
class derived : public base
{
public:
void func() override
{
}
};
当我使用抽象类作为此类的模板参数(abstraction<base(derived)>
)时,我得到以下错误:
VC++ 2015.3 error: C2259: 'base': cannot instantiate abstract class
GCC 4.9.2 error: invalid abstract return type 'base'
我想知道它是编译器中的错误还是禁止使用这种语法的抽象类型?
Link来源代码。
答案 0 :(得分:3)
模板参数TAbstract(TDeriveds...)
是一种函数类型,声明一个函数将TDeriveds...
作为参数并按值返回TAbstract
。按值返回时,必须可以实例化值类型的对象。抽象类的对象不能实例化,即按值返回它们是非法的(将它们作为参数)。相关条款是10.4 [class.abstract]第3段:
抽象类不能用作参数类型,函数返回类型或显式转换的类型。 ...
但是,您可以通过引用返回抽象基类:
template< class TAbstract, class ...TDeriveds >
class abstraction< TAbstract&(TDeriveds...) >
// ...
可悲的是,这意味着该类用户在使用该课时需要提及&
。
答案 1 :(得分:1)
实际上,使用TAbstract&amp; (或指针类型)作为类的类型参数将解决问题。
问题很简单:你不能返回一个抽象类的实例。相反,您可以返回TAbstract类型的引用或指针。