当前位置: 首页 > news >正文

每日一个C++知识点|虚函数

C++虚函数

作为C++程序员,“多态”绝对是绕不开的核心知识点,而撑起多态的关键技术,正是今天要聊的虚函数

一、先搞懂:虚函数到底是啥?

一句话概括:基类中加了virtual关键字的成员函数,就是虚函数

它的核心“魔法”的是:当用基类指针指向子类对象时,调用这个函数会自动执行子类的版本,而不是基类的!

举个生活中的例子:

  • 基类就像“动物”,有个“叫”的行为;
  • 子类是“狗”“猫”,继承了“动物”,但“叫”的方式不一样(狗汪汪、猫喵喵);
  • 没有虚函数时:用“动物指针”指向“狗”,调用“叫”会输出“动物叫”(认指针类型);
  • 有了虚函数后:同样的指针指向“狗”,会输出“汪汪叫”(认对象真实身份)。

这就是虚函数的核心价值——让指针“认清”对象的真实面目,实现多态!

二、3步实现虚函数多态

光说不练假把式,咱们用代码一步步实现上面的“动物叫”例子,一看就懂~

第一步:定义基类,给函数加virtual

先创建“动物”基类,把“叫”函数设为虚函数(加virtual关键字):

#include<iostream>usingnamespacestd;// 基类:动物classAnimal{public:// 虚函数:动物叫virtualvoidmakeSound(){cout<<"动物发出叫声~"<<endl;}};

第二步:定义子类,重写虚函数

创建“狗”和“猫”两个子类,继承Animal,并重写makeSound函数。这里可以加override关键字(可选,但强烈推荐),明确表示“我在重写基类的虚函数”,避免写错参数/返回值:

// 子类:狗(继承动物)classDog:publicAnimal{public:// 重写虚函数,加override更规范voidmakeSound()override{cout<<"汪汪汪!小狗在叫~"<<endl;}};// 子类:猫(继承动物)classCat:publicAnimal{public:voidmakeSound()override{cout<<"喵喵喵!小猫在叫~"<<endl;}};

第三步:测试多态效果

用基类指针指向不同的子类对象,调用makeSound,看看会不会自动匹配对应版本:

intmain(){// 基类指针Animal*animalPtr;// 指向狗对象Dog dog;animalPtr=&dog;animalPtr->makeSound();// 输出:汪汪汪!小狗在叫~// 指向猫对象Cat cat;animalPtr=&cat;animalPtr->makeSound();// 输出:喵喵喵!小猫在叫~return0;}

运行结果完全符合预期!同一个基类指针,指向不同子类对象时,自动调用子类的函数——这就是虚函数实现的多态,是不是很简单?

三、必记避坑点:3个关键规则

学会用法还不够,这3个坑一定要避开,否则容易出bug或内存泄漏!

1. 析构函数必须设为虚函数

如果基类指针指向子类对象,删除指针时如果析构函数不是虚函数,会只调用基类的析构函数,子类的资源释放不掉,导致内存泄漏

正确做法:基类析构函数加virtual:

classAnimal{public:virtualvoidmakeSound(){...}// 虚析构函数virtual~Animal(){cout<<"动物基类析构"<<endl;}};classDog:publicAnimal{public:voidmakeSound()override{...}~Dog()override{cout<<"小狗子类析构"<<endl;}};// 测试:删除基类指针指向的子类对象Animal*ptr=newDog();deleteptr;// 会先调用Dog析构,再调用Animal析构,无内存泄漏

2. 这两个函数不能是虚函数

  • 构造函数:对象还没创建完成,虚函数表还没初始化,无法实现虚函数效果;
  • 静态函数:静态函数属于类,不属于对象,而虚函数依赖对象的真实类型,冲突。

3. 用override明确重写,避免笔误

重写时如果参数类型、个数、返回值写错(比如把void makeSound()写成void makeSound(int)),编译器不会报错,而是当成新函数——加了override后,编译器会检查是否真的重写了基类虚函数,有错直接提示!

四、一句话总结

虚函数的核心就是:用virtual关键字标记基类函数,让子类能重写,最终实现“基类指针指向子类对象时,调用子类函数”的多态效果

掌握虚函数,你就打通了C++多态的“任督二脉”,后续学习继承、接口设计等知识点都会更轻松~

http://www.rkmt.cn/news/118302.html

相关文章:

  • 地表最强 AI——Gemini?0 基础带你真正入门
  • Shipit自动化部署终极指南:5个核心功能彻底掌握
  • PyPTO算子框架:解决千亿参数DeepSeek-V3.2-Exp推理性能瓶颈的终极方案
  • 如何实现高性能文本选择:从交互事件到渲染优化
  • 算法-排序-10
  • 当 Gemini 3 + Nano Banana Pro 抹平了人类最后一丝优越感
  • 浅析NCE0130KA在功率开关设计中的应用特性
  • LSPosed框架升级指南:从传统Xposed到现代化模块开发的完美过渡
  • 3步搞定媒体服务器集成:Homepage实战配置指南
  • KolodaView完整贡献指南:从零开始参与iOS卡片滑动开源项目开发
  • GeoTools:构建下一代地理信息系统的终极解决方案
  • 终极B站视频下载神器:bilidown让你轻松收藏8K超清内容
  • 全球化产品本地化架构深度解析:从技术实现到文化适配
  • DeeplxFile:免费跨平台文件翻译工具的完整使用指南
  • RuoYi-Cloud-Plus工作流引擎终极指南:5分钟实现流程自动化
  • AVL-CRUISE电动汽车仿真:从入门到精通的完整指南
  • Qwen3-4B-FP8模型:从零开始的AI伙伴部署实战
  • 哔哩下载姬DownKyi终极指南:简单高效获取B站优质内容
  • 多任务调度终极指南:从并发控制到性能优化的完整解析
  • 19、Linux文本编辑与办公软件使用指南
  • Redisson Docker环境DNSMonitor日志优化终极方案
  • 高效服务器监控:5步快速定位性能问题的终极指南
  • 大专生玩转AI营销:当市场思维撞上人工智能,我们如何化解跨界冲突?
  • 探索AI图像修复新境界:浏览器端智能修复工具深度体验
  • OpenUSD工具链深度解析:从入门到精通的完整指南
  • 象牙塔外的算法革命:时间与金钱双重压力下,学生如何低成本破局数字经济?
  • 20、OpenOffice.org软件安装与使用指南
  • 后台开发看过来:这次带你一举拿下网络IO模型
  • 100 万行文本挑战(1 Million Lines File Processing Challenge)
  • Java Spring框架:从入门到进阶的十个核心维度