『 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++ 类构造函数初始化列表
构造函数初始化列表以一个冒号开始,接着是以逗号分隔的数据成员列表,每个数据成员后面跟一个放在括号中的初始化式。
42 30
|
16天前
|
C++
C++(十一)对象数组
本文介绍了C++中对象数组的使用方法及其注意事项。通过示例展示了如何定义和初始化对象数组,并解释了栈对象数组与堆对象数组在初始化时的区别。重点强调了构造器设计时应考虑无参构造器的重要性,以及在需要进一步初始化的情况下采用二段式初始化策略的应用场景。
|
16天前
|
存储 编译器 C++
C ++初阶:类和对象(中)
C ++初阶:类和对象(中)
|
16天前
|
C++
C++(十六)类之间转化
在C++中,类之间的转换可以通过转换构造函数和操作符函数实现。转换构造函数是一种单参数构造函数,用于将其他类型转换为本类类型。为了防止不必要的隐式转换,可以使用`explicit`关键字来禁止这种自动转换。此外,还可以通过定义`operator`函数来进行类型转换,该函数无参数且无返回值。下面展示了如何使用这两种方式实现自定义类型的相互转换,并通过示例代码说明了`explicit`关键字的作用。
|
16天前
|
存储 设计模式 编译器
C++(十三) 类的扩展
本文详细介绍了C++中类的各种扩展特性,包括类成员存储、`sizeof`操作符的应用、类成员函数的存储方式及其背后的`this`指针机制。此外,还探讨了`const`修饰符在成员变量和函数中的作用,以及如何通过`static`关键字实现类中的资源共享。文章还介绍了单例模式的设计思路,并讨论了指向类成员(数据成员和函数成员)的指针的使用方法。最后,还讲解了指向静态成员的指针的相关概念和应用示例。通过这些内容,帮助读者更好地理解和掌握C++面向对象编程的核心概念和技术细节。
|
16天前
|
存储 C++
C++(五)String 字符串类
本文档详细介绍了C++中的`string`类,包括定义、初始化、字符串比较及数值与字符串之间的转换方法。`string`类简化了字符串处理,提供了丰富的功能如字符串查找、比较、拼接和替换等。文档通过示例代码展示了如何使用这些功能,并介绍了如何将数值转换为字符串以及反之亦然的方法。此外,还展示了如何使用`string`数组存储和遍历多个字符串。
|
1月前
|
C++ 容器
C++中自定义结构体或类作为关联容器的键
C++中自定义结构体或类作为关联容器的键
32 0
|
1月前
|
存储 安全 编译器
【C++】类和对象(下)
【C++】类和对象(下)
【C++】类和对象(下)
|
29天前
|
存储 算法 编译器
c++--类(上)
c++--类(上)
|
1月前
|
编译器 C++
virtual类的使用方法问题之C++类中的非静态数据成员是进行内存对齐的如何解决
virtual类的使用方法问题之C++类中的非静态数据成员是进行内存对齐的如何解决