news 2026/6/15 20:09:24

C++虚函数与普通函数区别

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
C++虚函数与普通函数区别

在 C++ 中,虚函数(virtual function)和普通函数(non-virtual function)在行为机制和用途上有重要区别。下面从多个维度进行对比分析:

一、核心区别概览

特性虚函数普通函数
动态绑定运行时多态(动态绑定)编译时绑定(静态绑定)
继承行为可被派生类重写(override)派生类同名函数会隐藏基类函数
内存开销有虚函数表(vtable)开销无额外开销
调用效率稍慢(需通过虚表指针查找)更快(直接调用)
关键字使用virtual声明无需特殊关键字

二、详细技术对比

1.绑定时机不同

classBase{public:voidnormalFunc(){cout<<"Base normal\n";}// 普通函数virtualvoidvirtualFunc(){cout<<"Base virtual\n";}// 虚函数};classDerived:publicBase{public:voidnormalFunc(){cout<<"Derived normal\n";}voidvirtualFunc()override{cout<<"Derived virtual\n";}};intmain(){Base*ptr=newDerived();ptr->normalFunc();// 输出: Base normal(静态绑定)ptr->virtualFunc();// 输出: Derived virtual(动态绑定)deleteptr;return0;}

2.继承中的行为差异

classBase{public:voidshow(){cout<<"Base show\n";}virtualvoiddisplay(){cout<<"Base display\n";}};classDerived:publicBase{public:// 隐藏基类的 show(),不是重写voidshow(){cout<<"Derived show\n";}// 重写虚函数voiddisplay()override{cout<<"Derived display\n";}};intmain(){Derived d;Base&b=d;d.show();// Derived showb.show();// Base show(静态绑定)d.display();// Derived displayb.display();// Derived display(动态绑定)}

3.内存结构差异

classWithVirtual{intx;public:virtualvoidfunc1(){}virtualvoidfunc2(){}};// 大小通常为 8(指针) + 4(int) + 对齐 = 16 字节(64位系统)classWithoutVirtual{intx;public:voidfunc1(){}voidfunc2(){}};// 大小通常为 4 字节

三、虚函数表机制

每个包含虚函数的类都有一个虚函数表(vtable),对象包含指向该表的指针(vptr):

classAnimal{public:virtualvoidspeak()=0;virtualvoideat(){cout<<"Animal eating\n";}voidbreathe(){cout<<"Animal breathing\n";}// 普通函数};classDog:publicAnimal{public:voidspeak()override{cout<<"Woof!\n";}voideat()override{cout<<"Dog eating\n";}};// 内存布局示意:// Dog对象: [vptr] -> [&Dog::speak, &Dog::eat]// 其他成员变量...

四、使用场景建议

使用虚函数的情况:

  • 实现多态:需要通过基类指针/引用调用派生类特定实现
  • 框架设计:设计可扩展的类层次结构
  • 接口定义:创建抽象基类(纯虚函数)

使用普通函数的情况:

  • 性能关键:避免虚函数调用开销
  • 不需要多态:函数行为在派生类中不需要改变
  • 工具函数:静态的辅助函数
  • 构造函数:构造函数不能是虚函数
  • 内联函数:虚函数通常不能被内联优化

五、代码示例:完整对比

#include<iostream>usingnamespacestd;// 基类classShape{protected:doublex,y;public:Shape(doublex,doubley):x(x),y(y){}// 普通函数:所有形状共享相同实现voidmove(doubledx,doubledy){x+=dx;y+=dy;cout<<"Shape moved\n";}// 虚函数:不同形状有不同实现virtualdoublearea()const{cout<<"Shape area (should override)\n";return0.0;}// 纯虚函数:强制派生类实现virtualvoiddraw()const=0;virtual~Shape(){}// 虚析构函数,确保正确释放资源};// 派生类classCircle:publicShape{doubleradius;public:Circle(doublex,doubley,doubler):Shape(x,y),radius(r){}// 重写虚函数doublearea()constoverride{return3.14159*radius*radius;}voiddraw()constoverride{cout<<"Drawing circle at ("<<x<<", "<<y<<") with radius "<<radius<<endl;}// 新增普通函数(不涉及多态)doublegetRadius()const{returnradius;}};intmain(){Circlecircle(10,20,5);Shape*shapePtr=&circle;// 普通函数:静态绑定circle.move(1,1);// Circle 对象调用shapePtr->move(1,1);// Shape 指针调用,相同结果// 虚函数:动态绑定cout<<"Circle area: "<<circle.area()<<endl;// Circle 实现cout<<"Via pointer: "<<shapePtr->area()<<endl;// Circle 实现// 纯虚函数circle.draw();shapePtr->draw();return0;}

六、重要注意事项

  1. 虚析构函数:如果类可能被继承且通过基类指针删除,基类必须有虚析构函数
  2. final 关键字:C++11 允许使用final禁止进一步重写
  3. override 关键字:C++11 推荐使用,明确表示重写虚函数
  4. 默认参数:虚函数的默认参数在编译时确定,建议避免在虚函数中使用默认参数
  5. 内联虚函数:虚函数可以是内联的,但多态调用时不会内联

总结

  • 虚函数是实现运行时多态的基础,通过虚函数表实现动态绑定
  • 普通函数在编译时确定,效率更高但缺乏多态性
  • 在设计类层次结构时,应根据是否需要多态行为选择使用虚函数
  • 性能敏感的场景应谨慎使用虚函数,避免不必要的开销
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/12 20:49:25

基于Java+SpringBoot+SSM旅游出行指南系统(源码+LW+调试文档+讲解等)/旅游攻略系统/出行规划工具/旅行指南平台/旅游信息服务系统/出行助手应用

博主介绍 &#x1f497;博主介绍&#xff1a;✌全栈领域优质创作者&#xff0c;专注于Java、小程序、Python技术领域和计算机毕业项目实战✌&#x1f497; &#x1f447;&#x1f3fb; 精彩专栏 推荐订阅&#x1f447;&#x1f3fb; 2025-2026年最新1000个热门Java毕业设计选题…

作者头像 李华
网站建设 2026/6/15 18:01:03

CnOpenData 美股上市公司资产负债表

本数据库提供全面、系统的美股上市公司财务报表数据&#xff0c;涵盖利润表、资产负债表、现金流量表及财务指标表四张核心数据表。数据包含公司代码、报告日期、指标类型、指标名称、指标数值等关键字段&#xff0c;并延伸至营业收入、净利润、每股收益、资产负债结构、现金流…

作者头像 李华
网站建设 2026/6/12 17:14:43

Node.js 后端架构的“隐秘角落”:从 Fastify 引擎到类型系统的博弈

在构建高性能 Node.js 服务&#xff08;尤其是基于 NestJS&#xff09;时&#xff0c;我们往往会遇到一些反直觉的现象&#xff1a;明明名字一样的类型却报错、明明配置了上传却收不到文件、明明锁定了版本却还要担心依赖树。本文将带你深入后端开发的“隐秘角落”&#xff0c;…

作者头像 李华
网站建设 2026/6/12 17:14:16

如何正确看待期货反向跟单策略?

很多刚入行期货反向跟单的朋友&#xff0c;第一反应就是 “这行业肯定暴利&#xff01;” 核心逻辑很直接&#xff1a;普通散户做期货大多长期亏损&#xff0c;反着他们的操作来&#xff0c;按理说肯定能赚钱。这个思路看似没毛病&#xff0c;甚至有点 “躺赚” 的意味&#xf…

作者头像 李华
网站建设 2026/6/12 17:07:50

基于Java+SpringBoot+SSM户外救援系统(源码+LW+调试文档+讲解等)/野外救援体系/户外应急系统/野外应急方案/户外搜救装备/户外救援技术/户外救援设备/野外救援平台

博主介绍 &#x1f497;博主介绍&#xff1a;✌全栈领域优质创作者&#xff0c;专注于Java、小程序、Python技术领域和计算机毕业项目实战✌&#x1f497; &#x1f447;&#x1f3fb; 精彩专栏 推荐订阅&#x1f447;&#x1f3fb; 2025-2026年最新1000个热门Java毕业设计选题…

作者头像 李华