『 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
相关文章
|
3月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
1月前
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
59 12
|
2月前
|
设计模式 安全 C++
【C++进阶】特殊类设计 && 单例模式
通过对特殊类设计和单例模式的深入探讨,我们可以更好地设计和实现复杂的C++程序。特殊类设计提高了代码的安全性和可维护性,而单例模式则确保类的唯一实例性和全局访问性。理解并掌握这些高级设计技巧,对于提升C++编程水平至关重要。
61 16
|
2月前
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
|
2月前
|
存储 编译器 C++
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。
|
2月前
|
安全 C++
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
157 6
|
2月前
|
编译器 C++
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
|
3月前
|
安全 编译器 C语言
【C++篇】深度解析类与对象(中)
在上一篇博客中,我们学习了C++类与对象的基础内容。这一次,我们将深入探讨C++类的关键特性,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载、以及取地址运算符的重载。这些内容是理解面向对象编程的关键,也帮助我们更好地掌握C++内存管理的细节和编码的高级技巧。
|
3月前
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)
|
4月前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
127 19

热门文章

最新文章