【C++】继承(下)

简介: 【C++】继承(下)

虚继承

为了解决菱形继承的二义性和数据冗余的问题,提出了虚继承

探讨一下虚继承是如何解决数据冗余和二义性的

不是虚继承版本


#include<iostream>
using namespace std;
class A
{
public:
    int _a;
};
 class B : public A
//class B : virtual public A
{
public:
    int _b;
};
 class C : public A
//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;
}


调用内存窗口

刚开始查看不是虚继承的类

由于A是B和C的父类,所以A类的_a在B类和C类中都存在

而B类本身有一个_b的成员变量

C类本身有一个_c的成员变量


虚继承版本

#include<iostream>
using namespace std;
class A
{
public:
    int _a;
};
//class B : public A
class B : virtual public A
{
public:
    int _b;
};
//class C : public A
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;
}

发现在监视下存在三份数据,而内存中只存在一份数据


发现相比于不是虚继承的版本,03 和04 上面多个一行地址存在


使用 地址48 7b 32 00 加上偏移量20 正好为 02 00 00 00 即A的地址

使用 地址54 7b 32 00 加上偏移量12 正好为 02 00 00 00 即A的地址


因为B类和C类中都存在A,存在数据冗余和二义性,所以把A的数据放入公共区域,既不放入B中,也不放入C中

通过偏移量来寻找这个公共的位置


在这两个表中,偏移量没有存到第一个位置,因为第一个位置是为以后的多态做准备的


使用虚继承是为了解决数据冗余的问题,但是调用内存发现整体变大了么?

感觉变大是因为A太小了,解决数据冗余和二义性,会增加了两个指针,只节省了一个A,即4个字节,相当于多消耗了4个字节

如果A变大,就不亏了,指向的空间忽略不计,会创建很多对象,每个对象都指向该空间,由大家共同分担,所以消耗可以忽略不计

感觉变大是因为A太小了,解决数据冗余和二义性,会增加了两个指针,只节省了一个A,即4个字节,相当于多消耗了4个字节

如果A变大,就不亏了,指向的空间忽略不计,会创建很多对象,每个对象都指向该空间,由大家共同分担,所以消耗可以忽略不计

例题

通过初始化列表去调用B这个类,而B类是继承父类A的,在调用B类的构造函数时,需要先调用A类的构造函数

初始化列表去调用C这个类,C类是继承父类A的,在调用C类的构造函数时,需要先调用A类的构造函数

初始化列表去调用A类,需要调用A类的构造函数

这样看似是调用了3次A的构造函数,但是实际上通过使用虚继承,所以只有一份A的数据,是单独调用的那份A数据


选择A ---- A B C D

初始化列表初始化的顺序跟出现的顺序无关,跟声明的顺序有关,谁先声明谁先初始化

谁先被继承,谁就先声明

先继承的A,后继承的B和C

继承和组合

public继承是一种is-a的关系,每个派生类对象都是一个基类对象

如 学生和人 ,学生是一个人

组合是一种has-a的关系,假设B组合了A,每个B对象中都有一个A对象

如 车和轮胎的关系 ,车有轮胎


关联关系指的是耦合度,继承关系更紧密一些 ,说明继承的耦合度高

B可以直接用A的3个成员

D可以直接用C的一个成员,间接用另外的两个成员


若把A类中的_a1改了,会影响B类

A改动保护可能影响B

若把C类中的_c1改了,不会影响D类

C改动保护和私有成员基本不影响D


设计:低耦合,高内聚

所以单个模块之间关联越低越好,这样一个模块出现问题,最大值程度上减少对另一个模块的影响

若两个类干的事是一样的,就把他们两个放在一起,若这两个类毫不相关,就不要合在一起


所以尽量使用组合去降低耦合度,但要用多态时就需要使用继承,多态是建立在继承之上的


相关文章
|
1月前
|
C++ 开发者
C++学习之继承
通过继承,C++可以实现代码重用、扩展类的功能并支持多态性。理解继承的类型、重写与重载、多重继承及其相关问题,对于掌握C++面向对象编程至关重要。希望本文能为您的C++学习和开发提供实用的指导。
54 16
|
30天前
|
编译器 数据安全/隐私保护 C++
【C++面向对象——继承与派生】派生类的应用(头歌实践教学平台习题)【合集】
本实验旨在学习类的继承关系、不同继承方式下的访问控制及利用虚基类解决二义性问题。主要内容包括: 1. **类的继承关系基础概念**:介绍继承的定义及声明派生类的语法。 2. **不同继承方式下对基类成员的访问控制**:详细说明`public`、`private`和`protected`继承方式对基类成员的访问权限影响。 3. **利用虚基类解决二义性问题**:解释多继承中可能出现的二义性及其解决方案——虚基类。 实验任务要求从`people`类派生出`student`、`teacher`、`graduate`和`TA`类,添加特定属性并测试这些类的功能。最终通过创建教师和助教实例,验证代码
48 5
|
3月前
|
编译器 C++ 开发者
【C++】继承
C++中的继承是面向对象编程的核心特性之一,允许派生类继承基类的属性和方法,实现代码复用和类的层次结构。继承有三种类型:公有、私有和受保护继承,每种类型决定了派生类如何访问基类成员。此外,继承还涉及构造函数、析构函数、拷贝构造函数和赋值运算符的调用规则,以及解决多继承带来的二义性和数据冗余问题的虚拟继承。在设计类时,应谨慎选择继承和组合,以降低耦合度并提高代码的可维护性。
42 1
【C++】继承
|
7月前
|
编译器 C++
【C++】详解C++的继承
【C++】详解C++的继承
|
4月前
|
安全 程序员 编译器
【C++篇】继承之韵:解构编程奥义,领略面向对象的至高法则
【C++篇】继承之韵:解构编程奥义,领略面向对象的至高法则
112 11
|
4月前
|
C++
C++番外篇——对于继承中子类与父类对象同时定义其析构顺序的探究
C++番外篇——对于继承中子类与父类对象同时定义其析构顺序的探究
72 1
|
4月前
|
C++
C++番外篇——虚拟继承解决数据冗余和二义性的原理
C++番外篇——虚拟继承解决数据冗余和二义性的原理
63 1
|
4月前
|
安全 编译器 程序员
C++的忠实粉丝-继承的热情(1)
C++的忠实粉丝-继承的热情(1)
33 0
|
4月前
|
编译器 C++
C++入门11——详解C++继承(菱形继承与虚拟继承)-2
C++入门11——详解C++继承(菱形继承与虚拟继承)-2
55 0
|
4月前
|
程序员 C++
C++入门11——详解C++继承(菱形继承与虚拟继承)-1
C++入门11——详解C++继承(菱形继承与虚拟继承)-1
60 0