码迷,mamicode.com
首页 > 编程语言 > 详细

C++中虚继承类构造函数的正确写法

时间:2015-08-26 22:00:08      阅读:202      评论:0      收藏:0      [点我收藏+]

标签:

最近工作中某个软件功能出现了退化,追查下来发现是一个类的成员变量没有被正确的初始化。这个问题与C++存在虚继承的情况下派生类构造函数的写法有关。在此说明一下错误发生的原因,希望对更多的人有帮助。

我们代码中存在虚继承的类的继承结构与下图类似,并不是教科书中经典的菱形结构。从 Intermediate1 和 Intermediate3 到Base2 的继承都是虚继承。Base1 和 Base2 都包含一些成员变量,并提供了相应的构造函数接受指定的初始化值。Base2 还有一个缺省构造函数,把其成员变量都初始化为0。Intermediate1,2,3 也都提供了一个构造函数接受指定的初始化值,并在在初始化列表里调用Base1和Base2的构造函数完成初始化。

技术分享

一位同事在做重构时,不小心把Final的代码改成了:

class Final : public Intermediate2, public Intermediate3 {
public:
    Final (int a, int b, int c)
        : Intermediate2(a, b, c),
          Intermediate3(b, c)
    {

    }

};
class Intermediate1 : public Base1, virtual public Base2 {
public:
    Intermediate1(int a, int b, int c)
        : Base1(a), 
          Base2(b, c)
    {

    }
};

class Intermediate2 : public Intermediate1 {
public:
    Intermediate2(int a, int b, int c)
        : Intermediate1(a, b, c),
          Base2(b, c)
    {

    }
};

class Intermediate3 : virtual public Base2 {
public:
    Intermediate3(int b, int c)
        : Base2(b, c)
    {

    }
};

看上去,Final的构造函数将调用Intermediate2 和 Intermediate3的构造函数分别将m_a, m_b 和 m_c初始化成指定的值。可是,运行时发现m_b和m_c的值是0!明显,这是调用了Base2的缺省构造函数。

原来,C++的规则是:如果在继承链上存在虚继承的基类,则最底层的子类要负责完成该虚基类部分成员的构造。我们可以显式调用虚基类的构造函数完成初始化。如果不显式调用虚基类的构造函数,则编译器会调用虚基类的缺省构造函数。如果不显式调用虚基类的构造函数,而虚基类没有定义缺省构造函数,则会出现编译错误。

很多时候,对于继承链上的中间类,我们也会在其构造函数中显式调用虚基类的构造函数,因为一旦有人要创建这些中间类的对象,我们也要保证它们得到正确的初始化。

所以,如果我们要把m_b和m_c初始化成指定的值,Final的构造函数的正确写法应该是这样:

    Final (int a, int b, int c)
        : Base2(b, c),
          Intermediate2(a, b, c),
          Intermediate3(b, c)
    {

    }

完整的测试程序如下所示,有兴趣的同学可以自行编译运行一下。也可以在调试器中单步运行Final的构造函数,看看前后两种写法分别是调用了Base2的哪个构造函数。

#include "stdafx.h"
#include <iostream>

using namespace std;

class Base1 {
public:
    Base1(int a): m_a(a) {}

protected:
    int m_a;
};

class Base2 {
public:
    Base2(int b, int c): m_b(b), m_c(c) {}
    Base2() : m_b(0), m_c(0) {}

protected:
    int m_b;
    int m_c;
};

class Intermediate1 : public Base1, virtual public Base2 {
public:
    Intermediate1(int a, int b, int c)
        : Base1(a), 
          Base2(b, c)
    {

    }
};

class Intermediate2 : public Intermediate1 {
public:
    Intermediate2(int a, int b, int c)
        : Intermediate1(a, b, c),
          Base2(b, c)
    {

    }
};

class Intermediate3 : virtual public Base2 {
public:
    Intermediate3(int b, int c)
        : Base2(b, c)
    {

    }
};

class Final : public Intermediate2, public Intermediate3 {
public:
    Final (int a, int b, int c)
        : Base2(b, c),
          Intermediate2(a, b, c),
          Intermediate3(b, c)
    {

    }

    void Print() {
        cout<<m_a<<", "<<m_b<<", "<<m_c<<endl;
    }
};


int _tmain(int argc, _TCHAR* argv[])
{
    Final finalObj(1, 2, 3);
    finalObj.Print();

	return 0;
}

C++中虚继承类构造函数的正确写法

标签:

原文地址:http://www.cnblogs.com/kaige/p/4761699.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!