15

我对虚拟基类的工作方式有点困惑。特别是,我想知道如何调用基类的构造函数。我写了一个例子来理解它:

#include <cstdio>
#include <string>
using std::string;

struct A{
    string s;
    A() {}
    A(string t): s(t) {}
};

struct B: virtual public A{
    B(): A("B"){}
};

struct C: virtual public A {};

struct D: public B, public C {};

struct E: public C, public B {};

struct F: public B {};

int main(){
    D d;
    printf("\"%s\"\n",d.s.c_str());
    E e;
    printf("\"%s\"\n",e.s.c_str());
    F f;
    printf("\"%s\"\n",f.s.c_str());
    B b;
    printf("\"%s\"\n",b.s.c_str());
}

哪个输出

""
""
""
"B"

我不确定前两种情况会发生什么,但至少对于第三种情况,我期望输出为“B”。所以现在我很困惑。了解如何调用 A 的构造函数的规则是什么?

4

2 回答 2

10

始终只有一个构造函数调用,并且始终是您实例化的实际具体类。您有责任为每个派生类赋予一个构造函数,该构造函数必要时调用基类的构造函数,就像您在B的构造函数中所做的那样。

更新:很抱歉错过了你的主要观点!感谢 ildjarn。

但是,您实际上B继承自. 根据标准(FIDS 中的 10.1.4),“对于每个指定为虚拟的不同基类,最派生对象应包含该类型的单个基类子对象”。在您的情况下,这意味着在构造基础时,您的类会立即调用' 默认构造函数,而不是'。AFAB

于 2011-06-23T23:28:33.503 回答
7

虚拟基类始终由派生最多的类构造。

于 2011-06-23T23:34:40.733 回答