单继承与多继承
单继承:一个子类只一一个直接父类时称这个继承关系为单继承
多继承:一个子类有两个或者两个以上直接父类时称这个继承关系为多继承。
菱形继承:菱形继承是多继承的一种特殊情况。两个子类同时继承一个父类,又有一个子类继承这两个子类。
菱形继承的问题:从下面的代码可以看出,菱形继承有数据冗余和二义性的问题。
class Person{
public:
string _name;//姓名
};
class Student :public Person
{
protected:
int _num;//学号
};
class Teacher: public Person
{
protected:
int _id;//职工编号
};
class Assistant :public Student, public Teacher
{
protected:
string _majorCourse;//主修课程
};
void Test()
{
//这样会有二义性无法明确知道访问的是哪一个
Assistant a;
//a._name = "peter"; 报错不知道访问哪一个
//需要显示指定访问那个父类的成员可以解决二义性问题
//但是数据冗余问题无法解决
a.Student::_name = "xxx";
a.Student::_name = "yyy";
}
解决菱形继承的二义性和数据冗余的问题采用虚拟继承。即在继承时采用virtual关键字。虚拟继承不用在其他地方去使用.
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;
};
int main()
{
D d;
d.B::_a = 1;
d.C::_a = 2;
d._b = 3;
d._c = 4;
d._d = 5;
return 0;
}
通过以上代码我们发现:
通过观察内存发现,虚拟继承中基类部分在下,子类部分在上与普通继承是反的
B和C中存在两个类似于地址的数据,其实是两个指针,指向一张表。这两个指针叫虚基表指针,这两个表叫虚基表。虚基表中存在偏移量,通过偏移量可以找到A。虚基表中第一个存在的是相对于自己的偏移量,第二个是相对于派生类的起始位置的偏移量
普通单继承与虚拟继承的区别?
1.普通虚拟继承比普通的单继承多了4个字节
2.虚拟继承中基类部分在下,子类部分在上 与普通继承是反的
3.编译器会给派生类生成默认的构造函数-----------------要在对象的前4个字节放数据
4.对象前4个字节中的内容------------>指向一块内存空间(0,8)