【C++】菱形继承

我们先来看下菱形继承的基本视图以及基本的代码结构

下面来看下简单的代码以及数据结构:

 
  1. class Person

  2. {

  3. public:

  4. int a_p;

  5. };

  6. class Studen :public Person

  7. {

  8. public:

  9. int a_st;

  10. };

  11. class Stuff :public Person

  12. {

  13. public:

  14. int a_sf;

  15. };

  16. class st_sf :public Stuff, public Studen

  17. {

  18. public:

  19. int a_ss;

  20. };

  21. int main()

  22. {

  23. st_sf ss;

  24. return 0;

  25. }

上述代码简单的很,就是Student(学生)和stuff(工人)都继承了Person类,然后Student和stuff作为Person的派生类,但是他俩又作为了类st_sf 的两基类,在主函数中创建类st_sf 的对象,下面我们来看下创建的st_sf 对象的数据结构是咋样的?

 上图中说到的会导致一些问题,具体是什么问题呢,看下面代码:

 
  1. #include<iostream>

  2. using namespace std;

  3. class A

  4. {

  5. public:

  6. int val;

  7. public:

  8. A(int x = 0):val(x){}

  9. };

  10. class B : public A

  11. {

  12. public:

  13. int num;

  14. public:

  15. B(int x = 0) : num(x), A(x + 10) {}

  16. };

  17. class C : public A

  18. {

  19. public:

  20. int sum;

  21. public:

  22. C(int x = 10) :sum(x), A(x + 10) {}

  23. };

  24. class D :public B, public C

  25. {

  26. public:

  27. int total;

  28. public:

  29. D(int x = 10) :total(x), B(x + 10), C(x + 20) {}

  30. };

  31. int main()

  32. {

  33. D d(0);

  34. return 0;

  35. }

在上述代码中我们通过最底层基类A派生出来了B和C,然后B和C又共同派生出来了一个类D,通过实例化D对象来观察上述代码的数据结构视图:

我们可以清晰的看到,构建出来的D对象有歧义了B中的val值为20,C中的val值为30,很明显不符合正常情况,但是呢代码还是正常通过了,这是为什么呢??

上图中我们给出了产生这种现象的原因,就是因为上述现象,所以 才会有我们刚刚看到的歧义性,那么怎么解决这种问题呢???

我们在这里就给出来了一个解决方法:虚继承 

 虚继承会导致我们student和staff对象中存放与Person对象的指针偏移量,但是存放的不是指针,这样就可以避免我们的歧义性问题:

 
  1. class A

  2. {

  3. public:

  4. int val;

  5. public:

  6. A(int x = 0):val(x){}

  7. };

  8. class B :virtual public A

  9. {

  10. public:

  11. int num;

  12. public:

  13. B(int x = 0) : num(x), A(x + 10) {}

  14. };

  15. class C :virtual public A

  16. {

  17. public:

  18. int sum;

  19. public:

  20. C(int x = 10) :sum(x), A(x + 10) {}

  21. };

  22. class D :public B, public C

  23. {

  24. public:

  25. int total;

  26. public:

  27. D(int x = 10) :total(x), B(x + 10), C(x + 20), A(x + 100) {}

  28. };

  29. int main()

  30. {

  31. D d(0);

  32. return 0;

  33. }

我们看一下引入虚继承之后的代码运行是怎么样的:

 该结果和我们上述画的哪个图结构一样,而且也没有歧义性的结果出现!!!

最后再补充一点东西,不使用虚继承的菱形继承不光会有歧义性结果:

 
  1. //在没有引入虚继承之前,我们不可以这样调用:

  2. int main()

  3. {

  4. D d(0);

  5. d.val;//这样不可以调用,因为它的d对象的基类中有两个A中的成员变量,编译器不知道调用哪个

  6. //所以编译不通过

  7. }

会有val不明确的报错!!! 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
菱形继承指的是一个派生类继承自两个直接或间接基类,而这两个基类又共同继承自一个共同的基类,导致派生类中存在两份共同基类的数据成员,从而产生了命名冲突和二义性的问题。 解决菱形继承问题的一种方法是使用虚拟继承。虚拟继承可以使得共同基类在派生类中只有一份实例,从而避免了数据成员的重复和命名冲突问题。在使用虚拟继承时,需要在继承语句前加上关键字 virtual,例如: ``` class A { public: int a; }; class B : virtual public A { public: int b; }; class C : virtual public A { public: int c; }; class D : public B, public C { public: int d; }; ``` 在上面的例子中,B 和 C 都虚拟继承自 A,而 D 继承自 B 和 C。因此,在 D 中只有一份 A 的实例,从而避免了数据成员的重复和命名冲突问题。 在初始化菱形继承的派生类时,需要注意以下几点: 1. 派生类的构造函数必须调用每个直接基类的构造函数,以及虚拟基类的构造函数,顺序为先虚拟基类,再按照继承的顺序调用直接基类的构造函数。 2. 虚拟基类的构造函数由最底层的派生类负责调用,其他派生类不需要再次调用虚拟基类的构造函数。 3. 派生类的析构函数必须调用每个直接基类的析构函数,以及虚拟基类的析构函数,顺序为先按照继承的顺序调用直接基类的析构函数,再调用虚拟基类的析构函数。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值