菱形继承浅析

来源:互联网 发布:八位单片机 编辑:程序博客网 时间:2024/06/14 07:52

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个字节
虚继承的缺点:复杂,需寻址找到共有成员效率受影响。