C++菱形继承+虚继承

单继承和多继承
单继承:一个派生类只有一个直接基类。
(B的直接父类只有A,是单继承)

class AA
{
public:
    AA(int a=1)
    :_a(a)
    {
        cout << "AA()" << endl;
    }

    int _a;
};

class BB : public AA
{
public:
    BB(int b=2)
    :_b(b)
    {
        cout << "BB()" << endl;
    }

    int _b;
};

多继承:一个派生类的直接基类至少有两个。
(C类不仅继承了A类,还继承了B类,是多继承)

class CC : public AA ,public BB
{
public:
    CC(int c=3)
    :_c(c)
    {
        cout << "CC()" << endl;
    }

    int _c;
};

菱形继承属于多继承。

#include<iostream>
#include<stdlib.h>
using namespace std;

class A
{
public:
    A(int a=1)
    :_a(a)
    {
        cout << "A()" << endl;
    }

    int _a;
};

class B : public A
{
public:
    B(int b=2)
    :_b(b)
    {
        cout << "B()" << endl;
    }

    int _b;
};

class C : public A 
{
public:
    C(int c=3)
    :_c(c)
    {
        cout << "C()" << endl;
    }

    int _c;
};

class D : public B, public C
{
public:
    D(int d=4)
    :_d(d)
    {
        cout << "D()" << endl;
    }

    int _d;
};

int main()
{
    D d;
    d._a;
}

对上面代码进行编译,发现不通过,错误如下:

这里写图片描述

这个是菱形继承存在的二义性的问题。

这里写图片描述
从图中可以看出来,在D类中B类和C类分别存了一份A类的成员变量,造成了数据冗余的问题。

二义性问题可以通过指定类域进行访问,但是这样太麻烦,而且数据冗余的问题并没有得到解决。针对这个问题C++引入了虚继承。

虚继承:继承的前面加上virtual.
形如:class 派生类名:virtual 继承方式 基类名
(virtual为关键字,声明该基类为派生类的虚基类)

这里写图片描述

#include<iostream>
#include<stdlib.h>
using namespace std;

class A
{
public:
    A(int a=1)
        :_a(a)
    {
        cout << "A()" << endl;
    }

    int _a;
};

class B : virtual public A
{
public:
    B(int b=2)
        :_b(b)
    {
        cout << "B()" << endl;
    }

    int _b;
};

class C : virtual public A 
{
public:
    C(int c=3)
        :_c(c)
    {
        cout << "C()" << endl;
    }

    int _c;
};

class D : public B, public C
{
public:
    D(int d=4)
        :_d(d)
    {
        cout << "D()" << endl;
    }

    int _d;
};

int main()
{
    D d;
}

这里写图片描述
虚继承的内存分布有所变化,这里将公共的A类成员变量单独存起来,以前存A类成员变量的地方存了虚基类的地址,这个地址存的是A类成员变量距派生类地址的偏移量。

虽然虚继承解决了菱形继承二义性和数据冗余的问题,但是性能上会有所影响。

没有使用虚继承时,sizeof(d)=20;
使用虚继承后,sizeof(d)=24.

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C++菱形继承是指一个派生类同时继承了两个直接或间接基类,而这两个基类又间接或直接继承自同一个基类,从而形成了一个菱形的继承关系。 例如下面的代码: ``` 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; }; ``` 在这个例子中,类 `D` 继承了类 `B` 和类 `C`,而类 `B` 和类 `C` 都继承了类 `A`,因此形成了一个菱形继承关系。 菱形继承会引起一些问题,例如: 1. 内存浪费:由于类 `A` 被重复继承,导致在内存中存在两份相同的 `A` 对象,造成内存浪费。 2. 访问冲突:由于类 `D` 继承了类 `B` 和类 `C`,而这两个类都继承了类 `A`,因此在类 `D` 中访问 `A` 中的成员时会出现访问冲突的问题。 为了解决菱形继承带来的问题,可以使用虚继承虚继承可以解决内存浪费和访问冲突的问题,它的原理是在派生类中只保留一个虚基类的实例,由所有的派生类共享使用。 修改上面的例子,使用虚继承: ``` 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` 时使用了 `virtual` 关键字,表示使用虚继承。这样,类 `D` 中就只有一个 `A` 对象的实例,而且访问 `A` 中的成员也不会出现访问冲突的问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值