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