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++多重继承的情况下,子类中的虚函数表指针数量等于其含虚函数的基类数量,而虚函数表的数量则取决于子类是否新增了虚函数以及是否重写了继承自基类的虚函数。

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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

即插即用篇 | YOLOv8引入局部自注意力 HaloAttention | 为参数高效的视觉主干网络扩展局部自注意力

本改进已集成到 YOLOv8-Magic 框架。 我们提出了Axial Transformers,这是一个基于自注意力的自回归模型,用于图像和其他组织为高维张量的数据。现有的自回归模型要么因高维数据的计算资源需求过大而受到限制,要么为了减少资源需求而在分布表达性或实现的便捷性上做出妥协。相…

【知识碎片】2024_05_11

本篇记录了两个代码&#xff0c;【图片整理】是一个数组排序题&#xff0c;【寻找数组的中心下标】看起来很适合用双指针&#xff0c;但是细节多&#xff0c;最后还是没通过全部用例&#xff0c;看了题解写出来的。 C语言部分是两个知道错了之后恍然大悟的选择题。 每日代码 图…

1053: 输出利用先序遍历创建的二叉树中的指定结点的度

解法&#xff1a; c语言 #include<iostream> #include<vector> using namespace std; typedef struct tNodes{char val;tNodes* left, * right; }* tNode;void creat(tNode& t) {char ch;cin >> ch;if (ch #) t NULL;else {t new tNodes;t->val …

最少数量线段覆盖-华为OD

系列文章目录 文章目录 系列文章目录前言一、题目描述二、输入描述三、输出描述四、java代码五、测试用例 前言 本人最近再练习算法&#xff0c;所以会发布一些解题思路&#xff0c;希望大家多指教 一、题目描述 给定坐标轴上的一组线段&#xff0c;线段的起点和终点均为整数…

完整性验证器:迈向 Starknet 超高可扩展性的一大步

原文&#xff1a;https://www.starknet.io/en/content/the-integrity-verifier-a-leap-toward-starknet-hyperscaling&#xff1b;https://www.starknet.io/en/ecosystem/grant 编译&#xff1a;TinTinLand 核心观点 由 Herodotus 开发的完整性验证器&#xff0c;使开发者能够…

代码随想录算法训练营第六十三天| LeetCode84. 柱状图中最大的矩形

一、LeetCode 84. 柱状图中最大的矩形 题目链接/文章讲解/代码讲解&#xff1a;https://programmercarl.com/0084.%E6%9F%B1%E7%8A%B6%E5%9B%BE%E4%B8%AD%E6%9C%80%E5%A4%A7%E7%9A%84%E7%9F%A9%E5%BD%A2.html 状态&#xff1a;已解决 1.思路 这道题跟上道接雨水的题基本上是反…

【MySQL】锁

锁 全局锁 全局锁&#xff1a;对整个数据库实例加锁&#xff0c;加锁后整个实例就处于只读状态&#xff0c;其他语句都将被阻塞。 使用场景是&#xff1a;全库的逻辑备份 语法&#xff1a; 1、加全局锁 flush tables with read lock ;2、数据备份 mysqldump -uroot –pr…

【Web后端】web后端开发简介_Servlet简介

1.web后端开发简介 Java企业级开发&#xff0c;也就是学习]avaEE(Enterprise Edition)版本,是一种结构和一套标准。在应用中开发的标准就是Servlet、jsp和JavaBean技术。jsp技术现在已基本处于淘汰状态&#xff0c;简单了解即可web后端开发&#xff0c;基于B/S模式的开发体系。…

系分-历年论文题目

年份试题一试题二试题三试题四2023年信息系统数据转换与迁移敏捷开发方法论Devops及其应用论信息系统可行性分析2022年论原型法及其在信息系统开发中的应用论面向对象设计方法及其应用2021年论面向对象的信息系统分析方法论静态测试方法及其应用论富互联网应用的客户端开发技术…

13、FreeRTOS 事件标志组

文章目录 一、事件组(event group)的特性1.1 什么是事件标注组1.2 事件标注组的场景1.3 事件组的概念1.4 事件组的操作 二、事件组API2.1 创建2.2 删除2.3 设置事件2.4 等待事件2.5 同步点 一、事件组(event group)的特性 1.1 什么是事件标注组 事件标志位&#xff1a;表明某…

2024kali linux上安装java8

1 kali下载Java 8安装包 访问Oracle官网或其他可信的Java下载站点&#xff0c;如华为云的开源镜像站&#xff08;例如&#xff1a;https://repo.huaweicloud.com/java/jdk/8u202-b08/jdk-8u202-linux-x64.tar.gz&#xff09;。 确保下载的是与你的Kali Linux系统架构&#xf…

Collection工具类

Collection工具类的介绍 Collection 是一个操作Set、List和Map等集合的工具类Collection中提供了一些列静态的方法对集合元素进行排序、查询和修改的等操作 Collection的排序操作&#xff08;均为Static方法&#xff09; 1&#xff0c;reverse&#xff08;List&#xff09;&…

刷t2、、、

、、 public class ThisTest {public static void main(String args[]) {int i;for (;;) {System.out.println(1);}} } while()的循环条件等于for中循环条件。循环体会有一个条件改变等于for中类似自增条件。while()判断条件一般在while前面会初始化跟for中初始化一样。这样 w…

Linux入门攻坚——23、DNS和BIND基础入门1

DNS——Domain Name Service&#xff0c;协议&#xff08;C/S&#xff0c;53/udp&#xff0c;53/tcp&#xff09; BIND——Berkeley Internet Name Domain&#xff0c;ISC&#xff08;www.isc.org&#xff09; 互联网络上主机之间的通信依靠的是IP&#xff0c;而人或程序一般使…

GT2512-STBA 三菱触摸屏12.1寸型

T2512-STBA参数说明&#xff1a;12.1"、SVGA 800*600、65536色、TFT彩色液晶显示屏、AC电源、32MB内存 三菱触摸屏GT2512-STBA性能规格详细说明&#xff1a; [显示部] 显示软元件&#xff1a;TFT彩色液晶显示屏 画面尺寸&#xff1a;12.1寸 分辨率&#xff1a;SVGA 80…

Windows10环境搭建http服务器

我 的 个 人 主 页&#xff1a;&#x1f449;&#x1f449; 失心疯的个人主页 &#x1f448;&#x1f448; 入 门 教 程 推 荐 &#xff1a;&#x1f449;&#x1f449; Python零基础入门教程合集 &#x1f448;&#x1f448; 虚 拟 环 境 搭 建 &#xff1a;&#x1f449;&…

连锁收银系统的五大功能

连锁收银系统是零售行业中不可或缺的工具&#xff0c;它为连锁店铺提供了必要的管理和运营支持。一个完善的连锁收银系统应当具备以下五大功能&#xff0c;以满足不断发展的零售业务需求。 1. 进销存管理 进销存管理是连锁店铺运营的核心&#xff0c;也是连锁收银系统不可或缺…

每日两题 / 101. 对称二叉树 230. 二叉搜索树中第K小的元素(LeetCode热题100)

101. 对称二叉树 - 力扣&#xff08;LeetCode&#xff09; 用两个指针同时遍历树的左右子树即可 每次遍历时&#xff0c;一个指针向左&#xff0c;另一个就要向右。一个向右&#xff0c;另一个就要向左 /*** Definition for a binary tree node.* struct TreeNode {* in…

生信人写程序1. Perl语言模板及配置

生物信息领域常用语言 个人认为&#xff1a;是否能熟悉使用Shell(项目流程搭建)R(数据统计与可视化)Perl/Python/Java…(胶水语言&#xff0c;数据格式转换&#xff0c;软件间衔接)三门语言是一位合格生物信息工程师的标准。 生物信息常用语言非常广泛&#xff0c;我常用的有…

【C++】学习笔记——模板进阶

文章目录 十一、模板进阶1. 非类型模板参数2. 按需实例化3. 模板的特化类模板的特化 4. 模板的分离编译 未完待续 十一、模板进阶 1. 非类型模板参数 模板参数分为类型形参和非类型形参 。类型形参即&#xff1a;出现在模板参数列表中&#xff0c;跟在class或者typename之类的…