菱形继承问题和虚继承是如何解决二义性与数据冗余的

菱形继承问题和虚继承是如何解决二义性与数据冗余的

继承是c++的三大特性之一,其中菱形继承问题是一个值得我们学习和掌握的知识点。
1.什么是菱形继承呢?
菱形继承定义为:两个子类继承同一个父类,而又有子类同时继承这两个父类。
可能这样纯粹的文字大家不能直观的理解它,下面画一幅直观图便可一目了然:
这里写图片描述
菱形继承的代码描述如下:

#include<iostream>
using namespace std; 

class A
{
public:
    int _a;
};

class B : public A
{
public:
    int _b;
};

class C : public A
{
public:
    int _c;
};

class D :public B, public C
{
public:
    int _d;
};

void Test()
{
    D d;
    d.B::_a = 1;
    d.C::_a = 2;
}

int main()
{
    Test();
    return 0;
}

这里写图片描述
可以看到,在对_a赋值时,必须使用域访问限定符,否则无法识别是对BB对象还是对CC对象中的 _a赋值,虽然这样解决了二义性问题,但是又产生了数据冗余的问题.
2.如果要解决二义性和数据冗余的问题,则又需要引入虚继承的概念。
虚继承是在class B : public A与class C :public A的public前加上关键字virtual.
3.那么虚继承是如何解决二义性的呢?
要讨论这个问题,首先要来看看,对于菱形继承和虚继承来说, sizeof(b)的值是多大呢,菱形继承很容易看出答案是20个字节,很多人会想,那虚继承就是16字节了,但经过测试,发现应该是24字节,怎么会多出8个字节呢,这就涉及到内存的分配.
将Test()做修改:

void Test()
{
    D d;
    d.B::_a = 0;
    d.C::_a = 1;
    d._b=2;
    d._c=3;
    d._d=4;
}

菱形继承中对象d求sizeof(d)=20
这里写图片描述

可以看到内存分配正好符合菱形继承对象模型中变量的顺序
这里写图片描述
给class B : public A与class C :public A的public前加上关键字virtual后,变成菱形虚继承,现在来看虚继承的情况.
此时对象d的sizeof(d)=24
这里写图片描述

那么地址0X00BBF8CC和0X00BBF9D4中存储的是什么呢?
在去查看内存可以看到0X00BBF8CC和0X00BBF9D4分别存储了一个指针,该指针指向的内存偏移四个字节处存储的分别是十进制值20和12,通过观察0X00BBF8CC和0X00BBF9D4与0X00BBF8E0相差的字节数,刚好是20和12,因此,该值代表了偏移量

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
菱形继承(diamond inheritance)在多重继承中可能会引起一些问题。下面我将详细说明菱形继承存在的问题,并介绍继承的概念及其解决方案。 菱形继承问题: 1. 二义性(ambiguity):当派生类同时继承自两个或更多个基类,而这些基类又共同继承自一个公共基类时,会出现二义性。如果基类中有相同的成员函数或成员变量,编译器无法确定应该调用哪个基类中的成员,从而导致编译错误。 2. 冗余数据(redundant data):由于派生类继承了多个相同的基类,这些基类可能包含相同的成员变量。这样就会导致派生类中存在多份相同的数据,造成内存空间的浪费。 继承的概念及解决方案: 为了解决菱形继承所带来的问题,C++引入了继承(virtual inheritance)。继承通过在公共基类前加上`virtual`关键字来声明,在派生类中只保留一份公共基类的实例,从而避免了冗余数据二义性问题。 下面是一个示例代码,展示了如何使用继承解决菱形继承问题: ```cpp class Animal { public: int age; }; class Mammal : virtual public Animal { }; class Bird : virtual public Animal { }; class Platypus : public Mammal, public Bird { }; int main() { Platypus p; p.age = 5; // 直接访问公共基类的成员 return 0; } ``` 在上面的代码中,我们定义了一个`Animal`类作为公共基类,`Mammal`和`Bird`类通过继承分别继承自`Animal`类。然后,`Platypus`类通过多重继承同时继承自`Mammal`和`Bird`类。 通过使用继承,派生类`Platypus`中只保留了一份公共基类`Animal`的实例,避免了冗余数据。此外,我们可以直接访问公共基类`Animal`的成员变量,而不会出现二义性。 需要注意的是,继承可能会引入一些额外的开销,因为需要在内存中存储基类的指针或偏移量。此外,继承可能会导致一些设计上的复杂性,因此在使用继承时需要权衡利弊,并根据具体情况选择合适的解决方案。 希望这个解释能帮助你理解菱形继承存在的问题以及继承的概念和解决方案。如果你还有其他问题,请随时提问!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值