C++ | 动多态 | 虚函数表
多态机制
C++语言有三大特性:封装、继承、多态。
其中所谓的多态,即 “同一接口,不同形态”。接口在我们 C/C++ 语言中可以理解为函数名,不同形态可以理解为函数执行的功能不同。
而多态又主要分为静多态和动多态。
静多态: 在编译阶段确定函数的调用
动多态: 运行阶段确定函数的调用
宏多态: 在预编译阶段确定函数的调用
而我们今天要讨论的就是动多态。
动多态
提到动多态就要提到虚函数,其中虚函数的作用是为动多态提供支持。而我们说动多态发生在运行阶段,那么我们就要先了解运行阶段发生了哪些事情。
从一个 .c/.cpp 文件到最终的可执行程序,主要有以下四个步骤:预编译、编译、汇编和链接。

在链接成为可执行程序之后,就可以运行了,也就是我们所说的运行阶段。
虚函数表
下面我们采用推理的方式,一步步探究动多态的实现原理。
分析1:我们说动多态在运行阶段确认函数的调用 ===》而函数的调用需要拿到函数的入口地址 ===》
所有的文件都是存储在磁盘上的,而在运行阶段所有的数据都是在内存中,要想实现在运行时调用函数,就要确保内存中保存着这些函数的入口地址。那么问题来了,我们在运行中能不能拿到函数的入口地址呢?或者说,要通过怎样的方式才能拿到函数的入口地址呢?
要回答这个问题,我们就得先搞清楚链接阶段发生了哪些事情。
链接阶段主要做了以下四件事情:
- 合并段和符号表
- 符号解析
- 分配地址和空间
- 指令段(.text)符号重定位
函数的入口地址在符号表中存放,在链接完成之后我们就已经拿到了函数的入口地址,但是符号表在磁盘上存放 。
分析磁盘文件:
在windows系统上文件以PE格式存放,在Linux系统上文件以ELF格式存放,下面以Linux的ELF文件为例分析文件格式。
ELF文件格式大致如下,一个ELF HEADER头部信息、.text指令段、.data数据段、.bss数据段(不存在)、.comment 注释信息段等等。

使用命令 objdump -h obj.o查看目标文件的段布局

运行阶段,程序通过两个加载器把指令段和数据段信息加载到内存中,而在运行阶段我们根本就没有把符号表加载到内存中,也就拿不到函数的入口地址。(ps:函数在汇编阶段完成后,普通函数的入口地址已经写死到指令中了,因此在运行阶段不需要加载符号表。)
数据段有 .data段、.bss段、 .rodata段,以下为程序加载示意图

分析2:在内存中拿到函数的入口地址 ===》 程序运行时从可执行程序中只加载 指令和 数据 ===》 把函数的入口地址放在可加载的区域
问:可加载的区域就指令段和数据段,那么把入口地址放在指令段还是数据段呢?
答:嗯,地址应该算是数据吧,就如同指针所指向的数据一样,存放的都是地址。
那么最终的问题,就变成怎样把函数入口地址放入数据段中。
问:那么函数入口地址是哪个阶段产生的呢?
答:函数、全局变量、静态变量的虚拟地址在编译时就可确定。
编译阶段会生成函数符号和入口地址,按照普通函数的处理方式,函数符号会写进符号表中,而我们现在让它在编译阶段时在符号表放一份,在数据段再放一份。这样数据段中就有了一份函数入口地址的拷贝。其中数据段指的是 .rodata(只读数据段)。因此,在 .rodata 数据段中就有了一份符号表。
在这个 .rodata 段中存放的函数数据集合我们用一个结构体vftable表示,而这个函数集合也就是我们讨论的虚函数表。
通过层层推理,终于水落石出了。我们找出了多态实现的理论依据,即虚函数表的生成。
前面也提到了,虚函数为多态提供了支持,因此对于虚函数的处理应该满足以上的推论,下面让我们正向的推导一遍整个流程:
虚函数的处理流程:
virtual 关键字声明函数为虚函数 ===》 编译阶段生成符号把函数符号写入符号表并生成虚函数表 ===》 汇编、链接生成可执行程序 ===》 加载指令和数据 ,运行可执行程序
C++ | 动多态 | 虚函数表的更多相关文章
- C++ 虚函数表解析
转载:陈皓 http://blog.csdn.net/haoel 前言 C++中 的虚函数的作用主要是实现了多态的机制.关于多态,简而言之就是用父类型别的指针指向其子类的实例,然后通过父类的指针调用实 ...
- C++ 多态、虚函数机制以及虚函数表
1.非virtual函数,调用规则取决于对象的显式类型.例如 A* a = new B(); a->display(); 调用的就是A类中定义的display().和对象本体是B无关系. 2. ...
- C++迟后联编和虚函数表
先看一个题目: class Base { public: virtual void Show(int x) { cout << "In Base class, int x = & ...
- C++ 知道虚函数表的存在
今天翻看陈皓大大的博客,直接找关于C++的东东,看到了虚函数表的内容,找一些能看得懂的地方记下笔记. 0 引子 类中存在虚函数,就会存在虚函数表,在vs2015的实现中,它存在于类的头部. 假设有如下 ...
- C++虚函数和虚函数表
前导 在上面的博文中描述了基类中存在虚函数时,基类和派生类中虚函数表的结构. 在派生类也定义了虚函数时,函数表又是怎样的结构呢? 先看下面的示例代码: #include <iostream> ...
- C++ Daily 《5》----虚函数表的共享问题
问题: 包含一个以上虚函数的 class B, 它所定义的 对象是否共用一个虚函数表? 分析: 由于含有虚函数,因此对象内存包含了一个指向虚函数表的指针,但是这个指针指向的是同一个虚函数表吗? 实验如 ...
- C++虚函数表
大家知道虚函数是通过一张虚函数表来实现的.在这个表中,主要是一个类的虚函数的地址表,这张表解决了继承.覆盖的问题,其内容真是反应实际的函数.这样,在有虚函数的类的实例中,这个表分配在了这个实例的内存中 ...
- 对C++虚函数、虚函数表的简单理解
一.虚函数的作用 以一个通用的图形类来了解虚函数的定义,代码如下: #include "stdafx.h" #include <iostream> using name ...
- 深入理解C++虚函数表
虚函数表是C++类中存放虚函数的一张表,理解虚函数表对于理解多态很重要. 本次使用的编译器是VS2013,为了简化操作,不用去操作函数指针,我使用到了VS的CL编译选项来查看类的内存布局. CL使用方 ...
随机推荐
- 设计模式学习笔记(详细) - 七大原则、UML类图、23种设计模式
目录 设计模式七大原则 UML类图 设计模式分类 单例模式 工厂设计模式 简单工厂模式 工厂方法模式(使用抽象类,多个is-a) 抽象工厂模式(使用接口,多个like-a) 原型模式 建造者模式 适配 ...
- HarmonyOS方舟开发框架容器类API的介绍与使用
作者:liuxin,华为高级工程师 容器类,顾名思义就是存储的类,用于存储各种数据类型的元素,并具备一系列处理数据元素的方法.在方舟开发框架中,容器类采用了类似静态的语言来实现,并通过NAPI框架对外 ...
- JZ-005-用两个栈实现队列
用两个栈实现队列 题目描述 用两个栈来实现一个队列,完成队列的Push和Pop操作. 队列中的元素为int类型. 题目链接: 用两个栈实现队列 代码 import java.util.Stack; / ...
- 含变量的字符串拼接(string.Format()或$"")
含变量的字符串拼接,一般不要用 + 来连接字符串,可用以下两种方式: 一.string.Format() 二.$"" (在C#6以上的版本中可用,推荐这种写法) 1 public ...
- thinkPHP ajax 状态修改(上架修改为下架)
<td> {if $v.status==1} <span class="top{$v.id}" name="0" onclick=" ...
- 【故障公告】没有龙卷风,k8s集群翻船3次,投用双集群恢复
今天没有龙卷风(异常的高并发请求),故障却依然出现,问题非常奇怪. 某种异常情况会造成短时间内, k8s 集群中大量 pod (超过60%)因健康检查失败而处于 CrashLoopBackOff 状态 ...
- LGP3126题解
这道题还有点意思. 路径要求是一个回文串,回文串立马枚举中点.中点只可能在对角线上. 枚举对角线上的一个点,然后两边的路径必须完全相同. 既然路径上的字符必须完全相同,那么每个前缀也必须完全相同. 考 ...
- LGP5653口胡
操作好像比较神秘. 发现 \(k\) 很小,考虑和 \(k\) 有关的 DP,考虑不出来. 费用提前计算,对 \(w_i\) 做后缀和,那么序列的权值就是 \(\sum_{i=1}^nyw_i\). ...
- (acwing蓝桥杯c++AB组)1.2 递推
1.2 递推与递归 文章目录 1.2 递推与递归 位运算相关知识补充 pair与vector相关知识补充 题目目录与网址链接 下面的讲解主要针对这道题目的题解AcWing 116. 飞行员兄弟 - A ...
- ArcMap操作随记(6)
1.上流汇流区 [填洼]→[流向]→[分水岭] 2.输入坐标进行移动,也就是精确移动 [移动]工具(在自定义中,其中的[旋转]工具也有类似效果) 3.找最近的要素(矢量) [近邻分析]→[汇总] 4. ...