4

根据我对 C++ 继承的理解,每当调用子类的构造函数时,都会自动调用父类的构造函数。对于模板化的构造函数,模板参数的数据类型是自动推断的,即我们不需要单独指定模板参数。该程序生成一个我似乎不明白的编译错误。

#include <iostream>
#include <list>
#include <algorithm>

using namespace std;

class A{
  public:
    int x;
    int y;
    int first(){
      return x;
    }
    int second(){
      return y;
    }
};

class C{
  public:
    float a,b;
    C(){
      a = 0.0f;
      b = 0.0f;
    }
    template<class T>
      C(T t){
        a = t.first();
        b = t.second();
      }
};

class D: public C{
  public:
    float area(){
      return a*b; 
    }
}

int main(){
  A a;
  a.x = 6;
  a.y = 8;
  C c(a);
  D d(a);
  cout<<c.a<<" "<<c.b<<" "<<d.area()<<endl;
}

产生编译错误

test.cpp: In function ‘int main()’:
test.cpp:56:8: error: no matching function for call to ‘D::D(A&)’
test.cpp:56:8: note: candidates are:
test.cpp:44:7: note: D::D()
test.cpp:44:7: note:   candidate expects 0 arguments, 1 provided
test.cpp:44:7: note: D::D(const D&)
test.cpp:44:7: note:   no known conversion for argument 1 from ‘A’ to ‘const D&’

我不知道这里发生了什么。有任何想法吗?

4

2 回答 2

6

D必须将构造函数参数传递给C,因为您没有使用默认构造函数。

class D : public C {
public:
    template <typename T> D (T t) : C(t) {}
    float area () { /* ... */ }
};

错误的原因是您尝试D使用参数进行构造,但尚未声明任何允许您这样做的构造函数。此外,您必须将参数传递给C,否则编译器将使用C的默认构造函数。

编译器错误信息可以这样分析。

test.cpp:56:8: error: no matching function for call to ‘D::D(A&)’

编译器抱怨:

D d(a);

并且它无法弄清楚如何D在传递类型的东西时构造一个A

然后它显示了它知道的两种构造函数选择:

test.cpp:44:7: note: D::D()
test.cpp:44:7: note: D::D(const D&)

它指出,对于每一个,它都有一个不能使用它的原因。对于第一个,它不需要任何参数。对于第二个,它无法将 typeA转换为 type D

于 2012-06-11T18:05:16.677 回答
1

根据我对 C++ 继承的理解,每当调用子类的构造函数时,都会自动调用父类的构造函数。

注意:父类的构造函数会自动调用,参数与子类的构造函数相同。

至于手头的具体问题:没有为类 D 声明构造函数。您将免费获得一个默认构造函数和复制构造函数,但不是 C 类中基于模板的构造函数。构造函数不是继承的。

于 2012-06-11T18:32:05.400 回答