菱形虚拟继承

单继承

一个子类只有一个直接父类时称这个继承关系为单继承。

class Person
{
public:
    void setinfo(int _age, const char *_name)
    {
        age = _age;
        name = _name;
    }

protected:
    int age;
public:
    string name;
};

class Student:public Person
{
public:
    void setinfo(int _age, const char *_name, int _num)
    {
        Person::setinfo(_age, _name);
        num = _num;
    }
protected:
    int num;
};
多继承

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

class Student
{
protected :
   int _num ;
   string _name;
};
class Teacher : public Person
{
protected :

    int _id ;

};
class Assistant : public Student, public Teacher
{
protected :
    string _majorCourse ;
};
菱形继承

菱形继承就是对单继承和多继承的一个组合

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 ; // 主修课程
};

Student 和Teacher类是对Person类的单继承,而Assistant类则是多继承前两者。
这里写图片描述

lass A
{
public:
    int _a;
};

class B1:public A
{

public:
    int _b;
};
class B2 :public A
{
public:
    int _c;
};
class D :public B1, public B2
{
public:
    int _d;
};

void test()
{
    D d;
    //d._a= 0;
    d._b = 1;
    d._c = 2;
    d._d = 3;
}

int main()
{
    test();
    return 0;

}

这里写图片描述
这里写图片描述

这里写图片描述

但是由派生类模型我们可以看到,Assistant类中有两个Person成员,在对象d的内存中也可以看到由两个随机值,这就存在二义性和数据冗余的问题。

为了解决菱形继承中的二义性问题,我们引入了虚拟继承。

class A
{
public:
    int _a;
};

class B:virtual public A
{

public:
    int _b;
};

void test()
{
    B b;
    b._a = 1;
    b._b = 2;
}

int main()
{
    test();
    return 0;

}

从我们学习到的普通继承知识,我们知道,如果是普通继承的话,派生类对象d的大小应该为8,因为只有基类的成员变量_a和派生类自己的的成员_b,
这里写图片描述

这里写图片描述

并且和普通继承不同,普通继承派生类模型是基类在上,派生类在下,这里是基类在下,派生类在上,并且在前面多了四个字节的内容,我们把这个类容当作地址再次
这里写图片描述
在这个多了两个数,一个0一个8,我们在上面的代码中在派生类中添加一个变量,_b,

lass B:virtual public A
{

public:
    int _b;
    int _b1;
};

这里写图片描述

从这里我们可以看出,这个0是派生类对于自己的偏移量,因为不管你类的内容怎么改变这个0都不变,但是下面那个数字会随着你派生类的大小而改变,下面的数是派生类和基类的偏移量,我们把他叫做偏移量表格,

所以,在虚拟继承中,有一个指向偏移量表格的偏移地址,

#define _CRT_SECURE_NO_WARNINGS

#include <iostream>

using namespace std;

class B
{
public:
    int _b;
};

class C1 :virtual public B
{
public:
    int _c1;
};

class C2 :virtual public B
{
public:
    int _c2;
};

class D :public C1, public C2
{
public:
    int _d;
};

void Test1()
{
    D d1;
    cout << sizeof(D) << endl;
    d1._c1 = 0;
    d1._c2 = 1;
    d1._d = 2;
    d1._b = 3;
}

int main()
{
    Test1();
    system("pause");
    return 0;
}

这里写图片描述
这里有两个偏移地址,
0x012bcb04
这里写图片描述
0x012bcf70
这里写图片描述
一开始运行程序时遇到的_b不明确的问题也没有出现了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值