C++之多态(下)

ops/2024/9/23 10:39:04/

目录

多态的实现原理

多态的拓展 

单继承中的多态 

多继承中的多态


上期,我们学习了多态的基本概念,本期我们来学习多态的实现原理。

多态的实现原理

class Base
{
public:virtual void func1(){cout << "Base::func1()" << endl;}
private:int _a;char _ch;
};

对于上述Base这个类而言,sizeof(Base)有多大呢?

大多数人可能认为是8个字节,按照以往的知识,考虑了内存对齐的大小之后,看似是8个字节貌似也没有什么问题,我们不妨通过实践,来看一下这个Base有多大。

通过运行结果不难发现,Base类的大小竟然是12字节,这是为什么呢?我们通过创建了一个Base类对象通过监视窗口进一步查看。

 通过监视窗口,我们不难看出,虽然我们只定义了两个成员,但是实际上Base类中有三个成员,那么这个_vfptr成员变量是什么呢?

_vfptr我们称它为虚函数表指针,它指向了一个虚函数表,虚函数表就是一个虚函数指针数组,里面存储虚函数的地址,在子类虚函数表中,会先去存放从子类中继承下来的虚函数地址,然后再去存放自己类中生成的虚函数的地址。

那么有了这个虚函数表指针之后,我们是如何在底层实现多态的呢? 

代码如下。

class Base
{
public:virtual void func1(){cout << "Base::func1()" << endl;}
private:int _a;char _ch;
};class Child:public Base
{
public:virtual void func1(){cout << "Child::func1()" << endl;}
};int main()
{Base b;Child c;Base& b2 = b;b2.func1();Base& b3 = c;b3.func1();return 0;
}

运行结果如下。 

我们发现,上述代码实现了多态。通过图示为大家讲解原理。

    当我我们把b对象传给它的引用b2时,b2调用func1函数时,先会去找b的虚函数指针,然后通过虚函数指针找到b的虚函数表,然后在虚函数表中找到func1函数的地址然后去调用,这样就完成了调用父类Base类中的虚函数func1。

    当我们把c对象床位它的引用b3时,b3调用func1函数时,先会去找c的虚函数指针,然后通过虚函数指针找到c的虚函数表,然后在虚函数表中找到func1函数的地址然后去调用,这样就完成了调用子类Child类中的虚函数func1。

    简单来说,就是父类类的指针或者引用会根据传来的是子类还是父类对象,去对应的子类或者父类对象的虚函数表中去调用对应的虚函数,这便是多态的实现原理。

多态的拓展 

示例代码如下。

我们发现,同样的两个父类对象,他们的虚函数指针也是相同的。这其实也是多态的一个特点,就是同类的对象共用一张虚函数表,子类会继承父类的虚函数表,但是会对继承下来的虚函数表进行改写,所以虽然子类会继承父类的虚函数表,但是因为进行了改写,所以本质上子类的虚函数表和父类的虚函数表是两张不同的表。

这不由得产生了一个问题,虚函数表存放在哪里?栈里面吗?

当然不是,如果是栈里面,那么虚函数表的生命周期就随对象,太过麻烦,我们直接给出结论,虚函数表是与虚函数一眼个都是存放在代码段的。

单继承中的多态 

代码如下。

#include<iostream>
using namespace std;class Base
{
public:virtual void func1(){cout << "Base::func1()" << endl;}virtual void func2(){cout << "Base::func2()" << endl;}private:int _a;char _ch;
};class Child:public Base
{
public:virtual void func1(){cout << "Child::func1" << endl;}virtual void func3(){cout << "Child::func3" << endl;}
};int main()
{Base b;Child c;return 0;
}

 调试窗口如下。

我们发现子类的虚表中,存储从父类中继承下来的func2以及重写之后的func1,自己的func3函数其实也在虚表中,但是通过调试窗口看不见,只能通过内存观察。

 

多继承中的多态

代码如下。

#include<iostream>
using namespace std;class Base1
{
public:virtual void func1(){cout << "Base1::func1()" << endl;}virtual void func2(){cout << "Base1::func2()" << endl;}private:int _a;char _ch;
};
class Base2
{
public:virtual void func3(){cout << "Base2::func3()" << endl;}virtual void func4(){cout << "Base2::func4()" << endl;}private:int _a;char _ch;
};class Child:public Base1,public Base2
{
public:virtual void func1(){cout << "Child::func1" << endl;}virtual void func3(){cout << "Child::func3" << endl;}virtual void func5(){cout << "Child::Func5" << endl;}
};int main()
{Child c;return 0;
}

监视窗口如下。

多继承中,子类会继承父类中的两个虚表并进行改写。继承下来的父类1的虚表存放从父类1中继承下来的fun1和func2函数,同样的,继承下来的父类2的虚表中存放从父类2中继承下来的func3和func4函数。

那么问题来了,子类中的虚函数func5到底存储在那个虚表里呢,是父类1还是父类2,父类1为Base1,父类2为Base2,调试通过内存进行观察。

Base1虚表中的内容如下。

 

Base2虚表中的内容如下。 

 

子类首先继承了父类Base1,然后继承了父类Base2,所以我们得出了结论,多继承中,子类的虚函数存放在首先继承的父类的虚表里面。 

以上便是多态的所有内容。

本期内容到此结束^_^ 

 


http://www.ppmy.cn/ops/97638.html

相关文章

2024.8.21 作业

一个服务器和两个客户端聊天 代码&#xff1a; /*******************************************/ 文件名&#xff1a;server.c /*******************************************/ #include <myhead.h> #define SER_IP "192.168.2.7" // 服务器IP #define SER…

【鸿蒙学习】HarmonyOS应用开发者高级认证 - 一次开发,多端部署

一、学习目的 掌握鸿蒙的核心概念和端云一体化开发、数据、网络、媒体、并发、分布式、多设备协同等关键技术能力&#xff0c;具备独立设计和开发鸿蒙应用能力。 二、总体介绍 HarmonyOS 系统面向多终端提供了“一次开发&#xff0c;多端部署”&#xff08;后文中简称为“一…

Docker 打包容器

使用 Docker 打包容器是一种将应用程序及其依赖项打包成轻量级、可移植容器的方式。 Docker 容器可以在任何运行 Docker 的环境中执行&#xff0c;从而确保应用程序在不同环境中的一致性。 以下是使用 Docker 打包容器的基本步骤&#xff1a; 1. 安装 Docker 首先&#xff0…

JAVA IO之基础知识

简介 IO 即 Input/Output&#xff0c;输入和输出。数据输入到计算机内存的过程即输入&#xff0c;反之输出到外部存储&#xff08;比如数据库&#xff0c;文件&#xff0c;远程主机&#xff09;的过程即输出。数据传输过程类似于水流&#xff0c;因此称为 IO 流。IO 流在 Java…

雅菲奥朗 FinOps 认证培训:开启企业云财务管理转型之路

前言&#xff1a; 在当今快速变化的商业环境中&#xff0c;企业面临着前所未有的IT财务挑战。随着云计算和数字化转型的推进&#xff0c;传统的财务管理方式已经不能满足“企业上云”的需求。FinOps&#xff0c;即“云财务管理”应运而生&#xff0c;成为帮助企业实现IT财务流…

Kafka 性能为什么比 RocketMQ 好

Kafka 性能更好的原因 因为 kafka 零拷贝技术跟 RocketMQ 的不一样。 kafka 零拷贝技术使用的是 sendfileDMA scatter/gather 。只需要经过 2 次拷贝&#xff0c;2 次上下文切换RocketMQ 零拷贝使用的 mmap 内存映射&#xff0c;需要经过 3 次拷贝&#xff0c;4 次上下文切换…

Table API SQL之时区Time Zone详解

目录 数据类型 日期和时间(Date and Time) 日期(DATE) 时间(TIME) 时间戳(TIMESTAMP) 无时区时间戳 有时区时间戳 本地时区时间戳(TIMESTAMP_LTZ) 区间年月(INTERVAL YEAR TO MONTH) 区间日秒(INTERVAL DAY TO SECOND) 时区(Time Zone) TIMESTAMP类…

windows docker 执行apt-get 权限问题

今天在windows下安装的docker 部署的容器执行apt-get遇到权限问题 PS C:\Users\xiaok> docker exec -it jenkins sh $ apt-get update Reading package lists... Done E: Could not open lock file /var/lib/apt/lists/lock - open (13: Permission denied) E: Unable to l…