『 C++类与对象 』多继承与虚继承

简介: 『 C++类与对象 』多继承与虚继承



⌨️多继承的概念

多继承指的是一个派生类是由多个基类继承而来的;

而在生活当中也有类似的例子:番茄既可以是水果,也可以是蔬菜;

而在C++2.0的版本中,就提出了多继承的概念,多继承允许一个派生类是由多个基类继承而来;


语法 🖱️

class Teacher {
protected:
 int _id;//工号
};
class Student{//使用virtual关键字
protected:
 int _num;//学号
};
class Grad : public Teacher ,public Student{
protected:
 string _major;
};

以该类为例:

该类中出现了Teacher老师类类与Student学生类两个类;

同时举出了一个Grad研究生类代表Teacher类与Student两个类的派生类;

上述语法即为多继承;


⌨️棱形继承

虽然棱形继承一定程度提高了继承的多样性;

但随之而来的也是一定的问题;

以多继承为基础,同时也出现了棱形继承;

棱形继承是在多继承基础上产生的,假设一个基类拥有多个派生类,并在多次继承之后又将其若干个派生类(或者其子类)多继承了一个派生类即为棱形继承;

【一个标准的棱形继承】

以图来看棱形继承并不存在过多问题;

但棱形继承的本质问题为数据冗余以及存在数据的二义性;

假设有一个类为Person;

class Person{
public:
 string _name;
};

以该类继承出了两个派生类分别为TeacherStudent类;

class Teacher :  public Person{
protected:
 int _id;//工号
};
class Student :  public Person{//使用virtual关键字
protected:
 int _num;//学号
};

最后以上面的两个类作为基类再继承了一个派生类为Grad

class Grad : public Teacher ,public Student{
protected:
 string _major;
};

此时的Grad类中被继承的Person中的_name成员该从TeacherStudent中哪个类访问;

这就是所谓的二义性;

以图中34,35行为例,可以这么解决数据的二义性问题(使用域作用限定符使得数据指向一个类域之中);

但是对于类中的数据冗余是不可避免的;

这里的类中以内置类型(不存在在堆中开辟空间)为例,若是存在需要开辟空间将会是一个较高的数据冗余问题;

如果从上个例子不能很好的观察到所谓的数据冗余和二义性的问题,接下来我将再举一个例子,并以GDB调试的形式观察其中的问题;

假设存在一段代码:

class A {
  public:
    int _a;
};
class B : public A{
  public :
    int _b;
};
class C : public A{
  public :
    int _c;
};
class D : public B, public C{
  public :
    int _d;
};
void test_2(){
 D d;
 d.C::_a = 1;
 d.B::_a = 2;
 d._b = 3;
 d._c = 4;
 d._d = 5;
}

编译并使用-g选项生成一个可调试的可执行程序;

并在GDB调试中再test_2函数的最后一行处打上断点并运行,在该断点处打印此时对象d时的状态;

此时的d对象中存在两个A类中的属性,这就是数据冗余;


⌨️虚继承

当然在紧接着在该版本的后一个版本中更新了对于解决棱形继承的问题 —— 虚继承virtual

关键字virtual即为虚继承的关键字;

语法:

class A {
  public:
    int _a;
};
class B : virtual public A{//使用virtual关键字表示虚继承
  public :
    int _b;
};
class C : virtual public A{
  public :
    int _c;
};
class D : public B, public C{
  public :
    int _d;
};

虚继承是如何解决数据冗余和二义性的(不谈虚表概念)?🖱️

以上面的代码为例,此时一样采用GDB调试的方式进行观察;

class A {
  public:
    int _a;
};
class B : virtual public A{//使用virtual关键字表示虚继承
  public :
    int _b;
};
class C : virtual public A{
  public :
    int _c;
};
class D : public B, public C{
  public :
    int _d;
};
void test_2(){
D d;
d.C::_a = 1;
d.B::_a = 2;
d._b = 3;
d._c = 4;
d._d = 5;
}

由于对于GDB调试来说并不是很好观察,所以采用可能比较麻烦的方式对其进行调试;

打印出对应的地址并对其进行观察:

从图中的可以看出内存的大致分布;

d对象的首地址处可以看到存储了一个指针,这个指针所指向的位置为一个数值(偏移量);

可以看到在_c_b之间的内存中其中一处所存储的为一个指针,而该指针指向的位置也存储了一个指针,而这个指针正是存储偏移量0x00000010的,而通过这个内存的地址0x7fffffffe430+这个偏移量即可访问到这个虚继承体系中公共的成员;

当需要对虚继承中共有的成员数据进行操作时将以特定的形式对这个共有的数据进行访问;

同时在一般情况下请不要使用虚继承,同时也尽量不要使用棱形继承;
相关实践学习
阿里云图数据库GDB入门与应用
图数据库(Graph Database,简称GDB)是一种支持Property Graph图模型、用于处理高度连接数据查询与存储的实时、可靠的在线数据库服务。它支持Apache TinkerPop Gremlin查询语言,可以帮您快速构建基于高度连接的数据集的应用程序。GDB非常适合社交网络、欺诈检测、推荐引擎、实时图谱、网络/IT运营这类高度互连数据集的场景。 GDB由阿里云自主研发,具备如下优势: 标准图查询语言:支持属性图,高度兼容Gremlin图查询语言。 高度优化的自研引擎:高度优化的自研图计算层和存储层,云盘多副本保障数据超高可靠,支持ACID事务。 服务高可用:支持高可用实例,节点故障迅速转移,保障业务连续性。 易运维:提供备份恢复、自动升级、监控告警、故障切换等丰富的运维功能,大幅降低运维成本。 产品主页:https://www.aliyun.com/product/gdb
相关文章
|
2月前
|
编译器 C++ 开发者
【C++】继承
C++中的继承是面向对象编程的核心特性之一,允许派生类继承基类的属性和方法,实现代码复用和类的层次结构。继承有三种类型:公有、私有和受保护继承,每种类型决定了派生类如何访问基类成员。此外,继承还涉及构造函数、析构函数、拷贝构造函数和赋值运算符的调用规则,以及解决多继承带来的二义性和数据冗余问题的虚拟继承。在设计类时,应谨慎选择继承和组合,以降低耦合度并提高代码的可维护性。
36 1
【C++】继承
|
2月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
60 2
|
2月前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
110 5
|
2月前
|
存储 编译器 C++
【c++】类和对象(中)(构造函数、析构函数、拷贝构造、赋值重载)
本文深入探讨了C++类的默认成员函数,包括构造函数、析构函数、拷贝构造函数和赋值重载。构造函数用于对象的初始化,析构函数用于对象销毁时的资源清理,拷贝构造函数用于对象的拷贝,赋值重载用于已存在对象的赋值。文章详细介绍了每个函数的特点、使用方法及注意事项,并提供了代码示例。这些默认成员函数确保了资源的正确管理和对象状态的维护。
109 4
|
2月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
140 4
|
3月前
|
存储 编译器 对象存储
【C++打怪之路Lv5】-- 类和对象(下)
【C++打怪之路Lv5】-- 类和对象(下)
35 4
|
3月前
|
编译器 C语言 C++
【C++打怪之路Lv4】-- 类和对象(中)
【C++打怪之路Lv4】-- 类和对象(中)
33 4
|
3月前
|
存储 安全 C++
【C++打怪之路Lv8】-- string类
【C++打怪之路Lv8】-- string类
30 1
|
3月前
|
存储 编译器 C++
【C++类和对象(下)】——我与C++的不解之缘(五)
【C++类和对象(下)】——我与C++的不解之缘(五)
|
3月前
|
编译器 C++
【C++类和对象(中)】—— 我与C++的不解之缘(四)
【C++类和对象(中)】—— 我与C++的不解之缘(四)