0

0

c++中虚函数的作用_c++多态实现原理【详解】

尼克

尼克

发布时间:2026-03-05 13:19:36

|

278人浏览过

|

来源于php中文网

原创

虚函数是实现运行时多态的必要条件,需用virtual声明才能触发动态绑定;否则编译器仅按指针/引用的静态类型调用函数,即静态联编。

c++中虚函数的作用_c++多态实现原理【详解】

虚函数是实现运行时多态的必要条件

没有 virtual 关键字,哪怕函数签名完全一致,子类重写也不会触发动态绑定。编译器只看指针/引用的静态类型,直接调用该类型的函数——这就是“静态联编”。加了 virtual 后,调用才通过虚表(vtable)查实际对象的类型,跳转到对应函数地址。

常见错误现象:
• 父类函数没加 virtual,用 Base* 指向 Derived 对象,调用的仍是 Base::func()
• 析构函数没声明为 virtualdelete base_ptr 时子类部分不析构,引发资源泄漏

实操建议:
• 只要类设计为被继承、且预期多态使用,所有需重写的成员函数都应加 virtual
• 基类析构函数必须是 virtual,哪怕它是空实现
• C++11 起推荐在派生类重写函数后加 override,让编译器检查签名是否真能覆盖

vtable 和 vptr 是多态的底层支撑机制

每个含虚函数的类,编译器生成一张虚函数表(vtable),存该类所有虚函数的地址;每个该类对象开头隐式插入一个虚表指针(vptr),指向其所属类的 vtable。对象构造时,vptr 被初始化为当前类的 vtable 地址。

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

关键细节:
vtable 是类级别的,不是对象级别的,同一类所有对象共享一张表
• 多重继承下,对象可能有多个 vptr(分别对应不同基类子对象)
sizeof 一个含虚函数的类,通常比无虚函数版本大一个指针宽度(如 8 字节 on x64)

性能影响:
• 虚函数调用比普通函数多一次内存读取(查 vtable)和一次间接跳转,但现代 CPU 分支预测对此优化较好
• 禁止内联:编译器无法在编译期确定调用目标,所以 virtual 函数默认不能被内联(除非 devirtualize 成功,如 LTO 下的全程序分析)

Designs.ai
Designs.ai

AI设计工具

下载

纯虚函数强制接口契约,但不提供实现

virtual void func() = 0; 声明的是纯虚函数,含纯虚函数的类是抽象类,不能实例化。它只规定“子类必须实现这个函数”,不关心怎么实现。

使用场景:
• 定义通用接口,如 Shape::draw()Stream::read()
• 类中无需任何默认行为,避免误用基类默认实现
• 配合模板 + CRTP 实现静态多态时,可作为占位标记

注意点:
• 纯虚函数可以有定义(在类外写函数体),但只能通过作用域解析符显式调用,如 Base::func()
• 构造函数/析构函数里调用虚函数,不会动态绑定——此时对象还没完全构造或已开始析构,vptr 指向当前正在构造/析构的类的 vtable

虚函数调用失败的典型陷阱

最隐蔽的问题不是语法报错,而是语义失效:代码能编译运行,但没走预期路径。

容易踩的坑:
• 函数签名不一致:子类函数参数类型用 int,父类是 const int&,看起来像重写,实为重载,virtual 失效
• 返回类型协变没满足要求:仅允许返回类类型指针/引用,且派生关系与当前类继承一致,否则编译报错
• 对象切片(slicing):用值传递接收派生类对象,传入基类形参时发生拷贝,丢失虚函数信息,后续调用全是基类行为
• 将局部对象地址赋给基类指针:函数返回后对象销毁,vptr 指向的内存已无效,调用虚函数导致未定义行为

调试建议:
• 在关键虚函数入口加日志或断点,确认实际执行的是哪个类的版本
• 用 gdb 查看对象内存布局:p/x *(void**)obj_ptr 可看到 vptr 值,再查其指向的 vtable 内容
• 开启编译器警告:GCC/Clang 的 -Wnon-virtual-dtor-Wsuggest-override 能提前暴露问题

虚函数机制本身简单,但它的生效依赖整个对象模型的协同:构造顺序、内存布局、指针类型、编译器优化层级都会影响最终行为。真正难的不是“怎么写”,而是“为什么没按想的走”。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

27

2025.11.27

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

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

557

2023.09.20

string转int
string转int

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

930

2023.08.02

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

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

603

2024.08.29

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

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

294

2025.08.29

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

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

212

2025.08.29

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

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

186

2023.11.23

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

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

125

2025.11.27

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

2

2026.03.05

热门下载

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

精品课程

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

共94课时 | 10.7万人学习

C 教程
C 教程

共75课时 | 5.2万人学习

C++教程
C++教程

共115课时 | 20.6万人学习

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

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