C++多态实现原理详解

阅读引言: 我想象了一下, 假如人有突然问我什么是多态, 我该如何给别人说清楚呢?所以写下这篇文章, 希望大家看完有所收获。

①. 开胃小菜

先看这样一个开胃小菜

这里我有点小小的疑惑, 大小为啥是1。

在C++中,结构体(struct)A的对象a的大小为1的原因可能是因为编译器对结构体进行了内存对齐。默认情况下,C++编译器会将结构体的起始地址与最大成员的对齐要求进行对齐。在这个例子中,结构体A是空的,没有任何成员,所以编译器将其大小设置为1字节,以满足内存对齐的要求。

结构体对齐, 大小为1, 1是不是能被任意的地址整除, 这是结构体对齐的知识。

②. 多态常见的一个小小面试题

假设使用上面的类实例化出一个类对象, 使用sizeof求该对象的大小?

结果为12个字节,这里使用的是32位环境。


为什么多出了四个字节呢?这多出来的四个字节是啥?为什么需要这个四个字节的空间, 用来干什么?


1,因为当一个类中出现虚函数的时候, 无论是自己本身的还是继承来的, 都会多出四个字节的空间, 这四个字节的空间其实是一个指针, 专业名词叫做虚函数表指针, 用来指向虚函数表。虚函数表指针(vptr)、虚函数表后面会介绍。

2, 这多出来的四个字节就是一个指针, 指向虚函数表, 这是编译器载编译期间帮我们做的, 伪代码如下。

3, 因为需要直到虚函数表的内存地址, 才能通过指针访问到虚函数, 虚函数表中的内容其实就是虚函数的入口地址。

③, 虚函数指针虚函数表

虚函数指针: 本质就是一个指针变量, 用来保存虚函数表的地址

虚函数表: 本质是内存中的一段连续空间, 空间内的每一项都是一个函数指针, 用来保存虚函数的入口地址。

内存布局: 

需要注意的是, 虚函数表属于类, 然后需要直到虚函数指针被赋值的时机是载钩爪函数中, 这是编译器默默为我们做的。

④. 多态的理解

代码层面上: 

多态存在的条件: 类中必须存在虚函数, 调用虚函数的方式必须使用虚函数表指针, 找到虚函数表, 接着调用里面的虚函数。换句话说就是必须是指针或引用调用的虚函数才是多态, 而静态创建的对象出现不了多态。

表现形式上看多态: 

第一条: 子类中重写父类中的虚函数是由要求的, 也就是函数的返回值、函数名、参数列表都需要相同, 但是这里有一个小小的特殊情况, 就是当基类中的虚函数返回的值基类指针或者引用的时候, 允许派生类中的函数的返回值返回派生类的指针或者引用。

#include <iostream> 
using namespace std; 
class A{};
class B:public A{}; 
class Base{ 
public: virtual void func(void){cout << "Base func" << endl; } virtual A* foo(void){cout << "Base foo" << endl; } 
}; class Derived: public Base{ void func(void) { cout <<"Derived func" << endl; }B* foo(void){                       //允许返回子类的指针或者引用,构成虚函数的覆盖条件 					cout << "Derived foo" << endl; }
}; int main(void){ Derived d1;Base *pd1 = &d1; pd1->func();Base& pd2 = d1;pd2.foo();return 0; 
} 

总结一下虚函数覆盖的条件: 

  • 只有类中的成员函数才能声明为虚函数,而全局函数、静态成员函数、构造函数都不能被声明为虚函数

  • 只有在基类中以virtual关键字声明的虚函数,才能作为虚函数被子类覆盖,而与子类中的virtual关键字无关

  • 虚函数在子类中的版本和基类中版本要具有相同的函数名,即函数名、参数表、常属性一致

  • 如果基类虚函数返回基本类型的数据,那么子类中的版本必须返回相同类型的数据;如果基类虚函数返回类类型指针(A)或引用(A&),那么允许子类中的版本返回其子类类型指针(B)或引用(B&)

我们来重点看一下第三句话

简单的看一下实现的原理 

当一个基类中存在虚函数的时候, 派生类机会从基类那里将其继承过来, 当派生类中函数满足基类中虚函数的覆写条件的时候, 就会将自己的虚函数表原先基类中的虚函数地址给替换成自己的函数地址, 这样, 不同的派生类只要是满足了虚函数的调用条件, 调用虚函数的时候, 调用的就是自己虚函数表中的那个自己实现的函数, 从而实现了多态。

从上面我们可以看出, 多态的实现其实就是将继承过来的虚函数表中原先的函数地址, 换成了自己的函数地址。

⑤, 问题

简单讲一下什么是多重继承, 就是一个类继承了多个基类。

在C++中,当进行多重继承时,子类中会有多个虚函数表指针,并且也会有多个虚函数表

首先,我们来理解虚函数表指针和虚函数表的概念。在C++中,如果一个类定义了至少一个虚函数,那么这个类就会拥有一个虚函数表(vtable),它存储了该类所有虚函数的地址。而虚函数表指针(vptr)是指向这个虚函数表的指针,它存在于每个拥有虚函数的类的实例中。

在多重继承的场景中,每个含有虚函数的基类都会为子类贡献一个虚函数表和对应的虚函数表指针。因此,如果一个子类从两个或多个含虚函数的基类继承而来,那么它就会拥有与这些基类数量相同的虚函数表指针。这些虚函数表指针的顺序与继承的顺序一致。

至于虚函数表的数量,如果子类没有新增虚函数,那么多重继承的子类会将全部基类的虚函数表继承下来。如果子类新增了虚函数,则这些新的虚函数会被添加到继承的第一个基类的虚函数表中,除非存在虚函数的重写,这种情况下新的虚函数会覆盖掉基类原有的虚函数

综上所述,在C++多重继承的情况下,子类中的虚函数表指针数量等于其含虚函数的基类数量,而虚函数表的数量则取决于子类是否新增了虚函数以及是否重写了继承自基类的虚函数。

好了, 以上就是全部内容, 图片资源部分来自网络, 如有侵权, 请联系我, 将其删除。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/10094.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Python | Leetcode Python题解之第74题搜索二维矩阵

题目&#xff1a; 题解&#xff1a; class Solution:def searchMatrix(self, matrix: List[List[int]], target: int) -> bool:row,col len(matrix),len(matrix[0])row_l,row_r 0,row-1while row_l < row_r:m (row_lrow_r)//2if target < matrix[m][0]:row_r m-1…

使用 Docker 部署 VS Code in The Browser

1&#xff09;介绍 GitHub&#xff1a;https://github.com/coder/code-server 在日常学习工作中&#xff0c;Vscode 已成为我们首选的代码编辑器。然而&#xff0c;其局限性在于当我们从家到公司移动时&#xff0c;难以保持连续的编码体验。针对这一痛点&#xff0c;虽然市面上…

oracle 数据库与服务、实例与SID、表空间、用户与表模式

一、数据库与数据库服务: 概念:就是一个数据库的标识,在安装时就要想好,以后一般不修改,修改起来也麻烦,因为数据库一旦安装,数据库名就写进了控制文件,数据库表,很多地方都会用到这个数据库名。是数据库系统的入口,它会内置一些高级权限的用户如SYS,SYSTEM等。我们…

11.买卖股票的最佳时机Ⅰ

文章目录 题目简介题目解答解法一&#xff1a;一次遍历代码&#xff1a;复杂度分析&#xff1a; 题目链接 大家好&#xff0c;我是晓星航。今天为大家带来的是 买卖股票的最佳时机面试题Ⅰ 相关的讲解&#xff01;&#x1f600; 题目简介 题目解答 解法一&#xff1a;一次遍历…

怎么把手机ip地址变成了外省

在日常使用中&#xff0c;有时我们可能因为某些原因需要快速切换手机的IP地址&#xff0c;特别是当需要从一个省份切换到另一个省份的IP时。这种需求可能来源于网络访问限制、地理位置相关服务的使用、或者网络安全等方面的考虑。那么&#xff0c;怎么把手机IP地址变成外省呢&a…

Adobe Premiere Pro v24.3.0 解锁版 (领先的视频编辑软件)

前言 Adobe Premiere Pro&#xff08;简称PR&#xff09;是一款知名的专业视频编辑软件&#xff0c;数字视频剪辑软件。主要用来编辑视频和音频&#xff0c;可以在RGB和YUV色彩空间中以高达32位色彩的视频分辨率对4K和更高质量的视频文件进行编辑&#xff0c;支持VST音频插件和…

C++ 搜索二叉树

目录 1.二叉搜索树概念 2. 实现二叉搜索树 2.1. 二叉搜索树的插入 2.2查找 2.3删除节点 3.二叉树的应用&#xff08;KV结构&#xff09; 1.二叉搜索树概念 二叉搜索树又称二叉排序树&#xff0c;它或者是一棵空树&#xff0c;或者是具有以下性质的二叉树: 若它的左子树不为…

Web界面加持!数据库备份神器,助你轻松备份数据!

使用这款带有Web界面的数据库备份神器&#xff0c;你可以轻松设置定时备份&#xff0c;确保数据安全无忧。备份结果即时通知&#xff0c;让你随时掌握备份状态。备份完成后&#xff0c;你将收到备份结果通知。无论是成功备份还是出现错误&#xff0c;你都能及时了解备份情况&am…

自适应熔断限流揭秘

原创 Chasen 拍码场 前言 自适应熔断与限流是在分布式系统中常用的机制&#xff0c;用于保护系统免受服务雪崩效应与突发流量影响。它能够根据系统的负载情况和性能指标自动调整限流策略&#xff0c;以确保系统能提供稳定可靠的服务&#xff0c;目前在业内已经有了不少的探索…

代码无界,创新无限!华为云开发者日 · 广州站来了!

5月23日&#xff0c;2024年首场华为云开发者日HDC.Cloud Day将在广州盛大举行。这场技术派对将为开发者们带来一场无与伦比的技术盛宴。在这里&#xff0c;开发者们将有机会现场聆听行业专家的精彩分享&#xff0c;深度了解众多前沿产品的最新技术和功能&#xff0c;并与行业专…

Sass语法介绍-运算

04 【Sass语法介绍-运算】 1.前言 运算是一种通过已知量可能的组合&#xff0c;获得新的量的行为。Sass 中也为我们提供了各种各样的运算&#xff0c;以便我们更好的处理样式代码。本节我们将学习 Sass 中的数字运算、关系运算、除法运算、颜色运算、字符串运算等等… 2.什么…

OSPF虚链路

原理概述 通常情况下&#xff0c;一个OSPF网络的每个非骨干区域都必须与骨干区域通过ABR路由器直接连接&#xff0c;非骨干区域之间的通信都需要通过骨干区域进行中转。但在现实中&#xff0c;可能会因为各种条件限制&#xff0c;导致非骨干区域和骨干区域无法直接连接&#x…

[muduo网络库]——muduo库三大核心组件之 Poller/EpollPoller类(剖析muduo网络库核心部分、设计思想)

接着上文&#xff0c;[muduo网络库]——muduo库三大核心组件之Channel类&#xff08;剖析muduo网络库核心部分、设计思想&#xff09;&#xff0c;本章我们来学习muduo网络库中第二大核心组件Poller/EpollPoller类。 先回顾一下三大核心组件之间的关系。 接着我们进入正题。 P…

解决VScode -正在本地下载 VS Code 服务器

不知道怎么回事再次连接服务器的时候一直卡在这里了&#xff0c;查看输出信息发现一直卡在下载处&#xff0c;报错信息如图1&#xff0c;输出信息如图2。 1.报错信息 图1 报错信息 图2 输出信息 2.尝试 【已解决】设置SSH主机&#xff1a;VS Code-正在本地下载 VS Code 服务器…

代码随想录算法训练营第二十五天 | 669. 修剪二叉搜索树、108.将有序数组转换为二叉搜索树、538.把二叉搜索树转换为累加树

669. 修剪二叉搜索树 题目链接/文章讲解&#xff1a; 代码随想录 视频讲解&#xff1a; 你修剪的方式不对&#xff0c;我来给你纠正一下&#xff01;| LeetCode&#xff1a;669. 修剪二叉搜索树_哔哩哔哩_bilibili 解题思路 在上一题的删除二叉树节点中&#xff0c;我们通过在…

无线收发模块家电控制实验

zkhengyang可申请加入数字音频系统研究开发交流答疑群(课题组) 当然可以先用固定电平发送&#xff0c;可以实现&#xff0c;0/1数据发送&#xff0c;接收。 可以使用51单片机来编码码&#xff0c;解码&#xff0c;或者任何MCU或者SOC&#xff0c;DSP&#xff0c;FPGA。 注意G…

初识指针(4)<C语言>

前言 前面的文章&#xff0c;已经对指针的基础概念以及运用有了初步了解&#xff0c;我们可以进一步探究指针比较深入的知识&#xff0c;下文将主要介绍&#xff1a;使用指针数组模拟二维数组、字符指针变量、数组指针、二维数组传参的本质、函数指针、typedef关键字等。 目录…

F. Circle Perimeter

思路&#xff1a;线性时间复杂度就可以解决&#xff0c;不用二分&#xff0c;我们枚举横坐标&#xff0c;然后看当前横坐标情况下多少个纵坐标满足条件。 代码&#xff1a; void solve(){int r;cin >> r;int y r, ans 0;for(int x 0;x < r;x ){ //枚举横坐标x&am…

只需3步,使用Stable Diffusion无限生成AI数字人视频(附安装包)

基本方法 搞一张照片&#xff0c;搞一段语音&#xff0c;合成照片和语音&#xff0c;同时让照片中的人物动起来&#xff0c;特别是头、眼睛和嘴。 语音合成 语音合成的方法很多&#xff0c;也比较成熟了&#xff0c;大家可以选择自己方便的&#xff0c;直接录音也可以&#…

Codeforces Round 944 (Div. 4) A - G

div.4只写部分题解了&#xff0c;都比较基础&#xff0c;数学偏多一点&#xff0c;几乎没有算法&#xff0c;有不懂的欢迎评论区提问&#xff01; A. My First Sorting Problem #include<bits/stdc.h> using namespace std ; typedef long long ll ; const int maxn 2…