菱形继承浅析

1:分析菱形继承的问题。
1.1三种继承关系下基类成员的在派生类的访问关系变化
1.2覆盖与隐藏
1.3单继承&多继承
1.4什么是菱形继承
2:剖析虚继承是怎么解决二义性和数据冗余的。
2.1菱形继承的所占内存大小
2.2为解决二义性及数据冗余,采用虚继承virtual

1.分析菱形继承的问题。
1.1三种继承关系下基类成员的在派生类的访问关系变化

public>protected>private,比较基类和继承关系,取其中较小者。父类为private则对子类不可见。

1.2覆盖与隐藏
子类与父类若有同名,则子类会直接覆盖父类。
访问时,默认访问子类,若要访问父类,则需域作用符。
析构函数即使不同名也会发生覆盖,由于在编译器中析构函数名相同。
析构时,先析构子类在析构父类(栈桢原因)。

#include<iostream>
using namespace std;

class person
{
public:
    int _name;
    void f(int)
    {
        cout << "f(int)" << endl;
    }
};

class student :public person
{
public:
    int _sno;
    void f()
    {
        cout << "f()" << endl;
    }
};
int main()
{
    student s;
    s.f();
    //s.f(5);
    s.person::f(5);
    return 0;
}

结果为
s.f()会直接调用student中的函数,若要访问person中的函数,则需域作用符。
若放开注释,s.f(5)会编译不通过,因其在父类中,被隐藏。
若给子类中f()更名为g(),则放开注释可编译通过,且显示为f(int)。

1.3单继承&多继承
单继承–一个子类只有一个直接父类时称这个继承关系为单继承
多继承–一个子类有两个或以上直接父类时称这个继承关系为多继承

1.4什么是菱形继承


Assistant中由两个person,存在二义性及数据冗余问题。

2.剖析虚继承是怎么解决二义性和数据冗余的。
2.1菱形继承的所占内存大小

#include<iostream>
using namespace std;

class person
{
public:
    int _name;
};

class student :public person
{
public:
    int _sno;
};

class teacher :public person
{
public:
    int _tid;
};

class assistant :public student, public teacher
{
public:
    int _major;
};

int main()
{
    assistant a;
    cout<<sizeof(a)<<endl;
    return 0;
}

结果为20

此时,int _name浪费了4个字节,若persong中有20个字节的共有成员,则浪费了20个字节。

2.2为解决二义性及数据冗余,采用虚继承virtual

#include<iostream>
using namespace std;

class person
{
public:
    int _name;
};

class student :virtual public person
{
public:
    int _sno;
};

class teacher :virtual public person
{
public:
    int _tid;
};

class assistant :public student, public teacher
{
public:
    int _major;
};

int main()
{
    assistant a;
    cout << sizeof(a)<<endl;
    return 0;
}

结果为24

注意virtual的位置

存储中实际只存了一份共有成员,只不过通过虚基表偏移寻址。
本文中虚继承要多出8个字节。若共同成员有100字节
不采用虚继承会浪费100字节
采用虚继承只浪费8个字节
虚继承的缺点:复杂,需寻址找到共有成员效率受影响。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值