深入C++之:一个类有几张虚函数表?

1. 没有任何虚函数的类:0 张

如果一个类没有声明任何虚函数,且它的所有基类也没有虚函数,那么编译器不会为这个类生成虚函数表。该类的对象内存中也不存在虚指针(vptr),完全是纯粹的数据成员集合。

2. 包含虚函数的基类 / 单继承:1 张

只要类中出现了虚函数,或者它通过单继承 自一个带有虚函数的基类,那么这个类就只有 1 张虚函数表。

  • 普通基类:编译器会为其生成 1 张虚表,存储所有虚函数的地址。该类的对象内部会悄悄插入 1 个虚指针(vptr)指向这张表。

  • 单重派生类:派生类会"拷贝"基类的虚表结构。

    • 如果派生类重写(override)了基类的虚函数,虚表中对应的函数地址会被替换为派生类自己的版本。

    • 如果派生类新增了独有的虚函数,这些函数的地址会被直接追加到这 1 张虚表的末尾。

3. 多重继承:多张(取决于有虚函数的基类数量)

在多重继承中,一个类会拥有多张 虚函数表。具体数量通常等于包含虚函数的基类的个数

假设类 Derived 继承自 Base1Base2(两者都有虚函数):

  • Derived 会拥有 2 张 虚函数表(分别对应 Base1 的子对象和 Base2 的子对象)。

  • Derived 实例化出的对象内存中,会包含 2 个虚指针(vptr),分别指向这 2 张不同的虚表。

  • 注意 :如果 Derived 自己新增了虚函数,按照主流编译器(如 GCC 和 MSVC)的实现,这些新增的虚函数地址通常会被追加到**第一个基类(Base1)**的虚函数表末尾,而不会去创建第 3 张表。

4. 虚继承(如菱形继承):情况复杂,包含 vbtable

当使用 virtual 关键字进行继承(为了解决多重继承带来的数据冗余问题)时,底层模型会发生本质变化:

  • 编译器会引入一种新的表:虚基类表(Virtual Base Table, vbtable),对象内部会增加指向该表的指针(vbptr),用于在运行时定位共享的虚基类子对象的位置。

  • 关于虚函数表(vtable)的数量:如果基类有虚函数,它依然有自己的虚表。但如果派生类在虚继承的同时又新增了虚函数,某些编译器(如 MSVC)为了内存对齐和安全,可能会为派生类额外单独生成 1 张新的虚函数表。这种情况下,虚表的数量会大于基类的数量。


总结

类的特征/继承结构 虚函数表 (vtable) 数量 对象中的虚指针 (vptr) 数量
无虚函数的普通类 0 0
包含虚函数的基类 1 1
单继承(基类有虚函数) 1 1
多重继承(N 个基类有虚函数) N N

注:C++ 标准并没有硬性规定虚函数表必须如何实现(这属于编译器 ABI 范畴),但主流编译器(GCC/Clang/MSVC)在处理虚表数量的逻辑上基本保持一致。

相关推荐
綝~6 小时前
爬虫数据采集工程师岗位面试题
爬虫·面试·请求
Qt程序员9 小时前
Linux RCU 原理与应用
linux·c++·内核·linux内核·rcu
qeen879 小时前
【C++】类与对象之类的默认成员函数(二)
android·c语言·开发语言·c++·笔记·学习
王老师青少年编程10 小时前
信奥赛C++提高组csp-s之搜索进阶(记忆化搜索案例实践3)
c++·记忆化搜索·方格取数·csp·信奥赛·csp-s·提高组
乐观的山里娃10 小时前
【反八股 01】HashMap 的设计参数是怎么来的
面试
嵌入式ZYXC11 小时前
第3篇:《面试题:I2C为什么要加上拉电阻?阻值怎么选?》
stm32·单片机·嵌入式硬件·面试·职场和发展
Titan202411 小时前
Linux动静态库
linux·服务器·c++
sbjdhjd11 小时前
面试(5)| 3.5 小时面试复盘第五弹:加班出差 + 客户响应 + 压力面全拆解
经验分享·程序人生·面试·职场和发展·开源·跳槽·求职招聘
j_xxx404_11 小时前
MySQL表操作硬核解析:从 CREATE TABLE 到磁盘文件、ALTER TABLE 与 DDL 风险
运维·服务器·数据库·c++·mysql·adb·ai
wuminyu11 小时前
Java锁机制之park和unpark源码剖析
java·linux·c语言·jvm·c++