0

0

C++ 怎么实现多态 C++虚函数与动态绑定机制详解【面试】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-27 11:13:09

|

864人浏览过

|

来源于php中文网

原创

基类指针调用虚函数时执行派生类版本,因编译器生成vtable并由对象vptr在运行时动态绑定;须通过指针或引用调用且函数声明为virtual,否则静态绑定。

c++ 怎么实现多态 c++虚函数与动态绑定机制详解【面试】

为什么基类指针调用函数时,实际执行的是派生类的版本

因为编译器在遇到 virtual 声明的函数时,会为该类生成虚函数表(vtable),每个对象头部隐式存储一个指向该表的指针(vptr)。运行时通过 vptr 找到对应函数地址,完成动态绑定。

关键前提是:必须通过指针或引用调用,且函数声明为 virtual。直接用对象值传递会触发静态绑定(即切片 + 编译期决议)。

  • virtual 函数:编译期根据变量静态类型决定调用哪个版本
  • virtual 函数:运行期根据对象实际类型查 vtable 决定调用哪个版本
  • 构造函数不能是 virtual;析构函数建议声明为 virtual(尤其基类有指针成员时)

虚函数表(vtable)在内存中长什么样

vtable 是编译器生成的静态数组,每个元素是函数指针,顺序与类中 virtual 函数声明顺序一致。单继承下,派生类 vtable 会复制基类部分,并覆盖被重写的函数地址;多继承则可能有多个 vptr,布局更复杂。

注意:vtable 不是对象的一部分,而是类级别的只读数据;vptr 才是每个对象开头的指针(通常 8 字节,在 64 位系统上)。

立即学习C++免费学习笔记(深入)”;

  • 可以用 sizeof 验证:带虚函数的类,即使无成员变量,sizeof 也至少为 8
  • gdb 中可打印 *((void**)obj) 查看 vptr 指向的首项(即第一个虚函数地址)
  • 纯虚函数在 vtable 中对应 nullptr,强制子类实现

override 和 final 关键字到底防什么

override 不是语法糖,它让编译器检查:当前函数是否真的重写了基类的 virtual 函数。拼错函数名、参数不匹配、const 修饰不一致都会报错。

阿里妈妈·创意中心
阿里妈妈·创意中心

阿里妈妈营销创意中心

下载

final 则禁止后续派生类再重写该函数,或禁止整个类被继承。两者都用于把本应在运行时暴露的问题(如意外未重写、误覆写)提前到编译期捕获。

  • 没加 override 时,看似重写成功,实则定义了一个新函数,基类虚函数仍按原逻辑走
  • 基类函数加了 final,子类里再写同签名函数并加 override,编译直接失败
  • 函数参数类型用引用/指针时,顶层 const 不影响重写判断;但底层 const(如 int* const)会影响

动态绑定失效的典型场景

最常见的是在构造函数和析构函数内部调用虚函数——此时动态绑定不生效,调用的是当前正在构造/析构的那个类的版本,而非最终派生类的版本。

原因:对象的 vptr 在构造函数执行过程中逐步被修改(先设为基类 vtable 地址,再逐层更新),析构时则逆向还原。所以中间状态无法反映完整类型。

  • 构造函数中调用 virtual 函数,等价于调用当前类的函数(哪怕派生类已重写)
  • 析构函数中同理,不会跳转到派生类实现
  • 避免在构造/析构中做依赖多态的行为;如需初始化逻辑,可提取为独立的 init() 并由用户显式调用

虚函数机制本身开销很小(一次指针解引用 + 偏移寻址),但真正容易出问题的地方,往往不是“怎么写”,而是“在哪调”和“谁来管生命周期”。尤其是跨模块传递对象、用智能指针管理多态对象时,vptr 的存在和析构顺序会直接影响行为是否符合预期。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
java多态详细介绍
java多态详细介绍

本专题整合了java多态相关内容,阅读专题下面的文章了解更多详细内容。

15

2025.11.27

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

529

2023.09.20

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

422

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

544

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

73

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

176

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

98

2025.11.27

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

6

2026.01.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
CSS3 教程
CSS3 教程

共18课时 | 4.9万人学习

Sass 教程
Sass 教程

共14课时 | 0.8万人学习

Pandas 教程
Pandas 教程

共15课时 | 1.0万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号