[C++从入门到精通] 14.虚函数、纯虚函数和虚析构(virtual)

  • 📢博客主页:https://blog.csdn.net/weixin_43197380
  • 📢欢迎点赞 👍 收藏 ⭐留言 📝 如有错误敬请指正!
  • 📢本文由 Loewen丶原创,首发于 CSDN,转载注明出处🙉
  • 📢现在的付出,都会是一种沉淀,只为让你成为更好的人✨

文章预览:

      • 一. 虚函数(virtual)
      • 二. 虚函数中的关键字
      • 三. 纯虚函数
      • 四*. 基类的析构函数务必写成虚函数(虚析构函数)
      • 五. 总结


一. 虚函数(virtual)

定义:在某基类中的成员函数:

  • 成员函数声明基类中为 virtual开头;
  • 该成员函数在一个或多个子类(派生类)中被重新声明、定义;

格式virtual 函数返回类型 函数名 ( 参数表 ) { 函数体 }

目的通过指向派生类的基类指针或引用,访问派生类中同名覆盖成员函数,实现多态性

  • 例如 Human *phumen = new Men(); //可通过基类Human的指针phumen调用子类中的同名函数,实现多态

多态性

  • 顾名思义就是“多个性态”。更具体一点的就是,用一个名字定义多个函数,这些函数执行不同但相似的工作。最简单的多态性的实现方式就是函数重载模板,这两种属于静态多态性。还有一种是动态多态性,其实现方式就是我们今天要说的虚函数

下面来看一段简单的代码:

class Human
{
public:
	void print() { cout << "This is 人类" << endl; }
};

class Men :public Human 
{
public:
	void print() { cout << "This is 男人" << endl; }
};

int main() 
{   
	Human human;
	Men men;
	human.print();
	men.print();
}

通过class Humanclass Menprint()这个接口,输出的结果也是我们预料中的,分别是This is 人类This is 男人

但这是否真正做到了多态性呢?

  • No多态还有个关键之处就是一切用指向派生类的基类指针或引用来操作对象。那现在就把main()处的代码改一改。
int main() 
{   
	//Human human;
	//Men men;
	//human.print();
	//men.print();
	Human * phuman = new Human;
	Human * phuman1 = new Men;
	phuman->print();
	phuman1->print();
}

在这里插入图片描述

可以看出,父类指针phuman1明明指向的是子类class Men对象但却是调用的父类class Humanprint()函数,这不是我们所期望的结果。

那么解决这个问题,即通过一个父类指针或对象调用所有子类中的成员函数或变量,就需要用到虚函数:

class Human
{
public:
	virtual void print() { cout << "This is 人类" << endl; }  //现在成了虚函数了
};

class Men :public Human
{
public:
	virtual void print() { cout << "This is 男人" << endl; } //这里需要在前面加上关键字virtual吗?
};

现在重新运行main的代码,这样输出的结果就是This is 人类This is 男人

在这里插入图片描述

毫无疑问,class A的成员函数print()已经成了虚函数,那么class Bprint()成了虚函数了吗?

回答是Yes,我们只需在把基类的成员函数声明前加virtual,其派生类的相应的同名同参成员函数也会自动变为虚函数。所以,class Bprint()也成了虚函数。对于在派生类的相应函数前是否需要用virtual关键字修饰, 看个人编程习惯。

总结:指向基类的指针在操作它的多态类对象时,会根据不同的派生类对象,调用其相应的函数,这个函数就是虚函数。‎


二. 虚函数中的关键字

override关键字

为了避免在子类中写错虚函数(没有和基类的成员函数同名同参),在C++11中,可以在子类虚函数声明后增加一个关键字 override

注意,override关键字用在子类中,而且是虚函数专用,用了这个关键字后,编译器会认为子类的虚函数覆盖了基类中的同名函数,那么编译器就会在父类中找同名同参的虚函数,如果没找到,编译器就会报错。这样,如果不小心在子类中把虚函数名称或参数写错了,编译器会帮助纠错。

final关键字

final关键字也是虚函数专用,但是是用在父类中的,作用是在父类的函数声明中加了final,那么任何尝试覆盖该函数的操作都将引发错误。


三. 纯虚函数

定义: 纯虚函数是在①基类中声明的虚函数,但它在基类中②没有定义,但③要求任何派生类都要定义自己的实现方法

格式: 在基类中实现纯虚函数的方法是在函数原型后加“=0” 

virtual void funtion1() = 0; //纯虚函数,在基类中定义,没有函数体,只有一个函数声明

抽象类由来:一旦基类中有纯虚函数,那么则不能生成这个类的对象,这个了就成为了“抽象类”。

抽象类目的:用来统一管理子类对象。

Human  human;                //不合法
Human *phuman = new Human;   //不合法

在这里插入图片描述

核心两点总结:

  • 含有纯虚函数的类叫抽象类,抽象类不能生成该类对象,主要用于当做基类来生成子类用的
  • 子类中必须要实现该基类中定义的纯虚函数;

问题:我们知道纯虚函数在基类中没有定义,那么虚函数在基类中一定要定义实现吗?

class Location
{
public:
	Location(){}
	~Location(){}

public:
	virtual bool Check();  // 这里一定要实现吗?
};

class LineLocation : public Location
{
public:
	LineLocation(){}
	~LineLocation(){}

public:
	virtual bool Check() {
		return 1;
	}
};

int _tmain(int argc, _TCHAR* argv[])
{
	Location* loc = NULL;
	loc = new LineLocation();
	bool b= loc->Check();
	return 0;
}

回答: 虚函数在基类中一定要实现,如果基类中的虚函数不想实现,只想通过派生类来实现,需要将基类中的虚函数换成纯虚函数(=0)。因为虚函数的地址在链接的时候需要放到类的虚函数表中,所以即使你的代码里面没有调用这个函数,编译器也需要取它的地址,已经有对它的引用了,就必须要实现才行。

注:因为纯虚函数就相当于接口,无法实例化,即Location loc;编译是不能通过的。即有纯虚函数的类,将其作为参数也好,另一个类的成员变量也好,只能将其定义为指针或引用,只要不给基类实例化对象就行。


四*. 基类的析构函数务必写成虚函数(虚析构函数)

基类中的虚拟成员希望其派生类定义自己的版本。特别是基类通常应该定义一个虚拟析构函数,即使它不起作用,析构函数必须是虚拟的,以允许动态分配和销毁继承层次结构中的对象。

那么为什么析构函数必须是虚拟的,而我们新建程序时,默认的析构函数却不是虚拟的呢?

1、为什么析构函数必须是虚拟的?

在这里插入图片描述
因为指针指向的是一个派生类实例,我们销毁这个实例时,肯定是希望先清理派生类自己的资源,同时又清理从基类继承过来的资源。而当基类的析构函数为非虚函数时,删除一个基类指针指向的派生类实例时只清理了派生类从基类继承过来的资源而派生类自己独有的资源却没有被清理

总结:如果一个类想要做基类(被其他类继承),那么我们必须定义这个类的析构函数并且还要将其写成虚函数(普通类可不定义析构函数为虚函数或直接不写析构函数)。这样,在delete释放指向的派生类实例的基类指针时,清理工作才能全面进行,才不会发生内存泄漏。

2、为什么默认的析构函数不是虚函数?

虚函数不同于普通成员函数,当类中有虚成员函数时,类会自动进行一些额外工作。这些额外的工作包括生成虚函数表虚表指针,虚表指针指向虚函数表。每个类都有自己的虚函数表,虚函数表的作用就是保存本类中虚函数的地址,我们可以把虚函数表形象地看成一个数组,这个数组的每个元素存放的就是各个虚函数的地址。
这样一来,就会占用额外的内存,当们定义的类不被其他类继承时,这种内存开销无疑是浪费的。

这样一说,问题就不言而喻了。当我们创建一个类时,系统默认我们不会将该类作为基类,所以就将默认的析构函数定义成非虚函数,这样就不会占用额外的内存空间。同时,系统也相信程序开发者在定义一个基类时,会显示地将基类的析构函数定义成虚函数,此时该类才会维护虚函数表和虚表指针。

参考博文:为什么析构函数必须是虚函数?为什么默认的析构函数不是虚函数?


五. 总结

1、定义纯虚函数是为了实现一个接口,起到一个规范的作用,规范继承这个类的程序员必须实现这个函数。
2、虚函数必须实现,如果不实现,编译器将报错。
3、调用虚函数执行的是“动态绑定”。动态:表示的就是在我们程序运行的时候才能知道调用了哪个子类的虚函数。
4、实现了纯虚函数的子类,该纯虚函数在子类中就编程了虚函数,子类的子类即孙子类可以覆盖该虚函数,由多态方式调用的时候动态绑定。
5、虚函数是C++中用于实现多态的机制。核心理念就是通过基类访问派生类定义的函数。
6、在有动态分配堆上内存的时候,析构函数必须是虚函数,但没有必要是纯虚的。
7、友元不是成员函数,只有成员函数才可以是虚拟的,因此友元不能是虚拟函数。但可以通过让友元函数调用虚拟成员函数来解决友元的虚拟问题。
8、析构函数应当是虚函数,将调用相应对象类型的析构函数,因此,如果指针指向的是子类对象,将调用子类的析构函数,然后自动调用基类的析构函数。


下雨天,最惬意的事莫过于躺在床上静静听雨,雨中入眠,连梦里也长出青苔。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/256492.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

APS54085 高辉度调光降压恒流芯片 PWM 线性调光 车灯IC

产品描述 APS54085 是一款 PWM 工作模式,简单、内置功率 MOS 管&#xff0c;适用于 5-100V输入的高精度降压 LED 恒流驱动芯片。电流2.0A。APS54085 可实现线性调光和 PWM 调光&#xff0c;线性调光有效电压范围 0.52-2.55V.PWM 调光频率范围 100HZ-30KHZ。APS54085 工作频率可…

MATLAB - Gazebo 仿真环境

系列文章目录 前言 机器人系统工具箱&#xff08;Robotics System Toolbox™&#xff09;为使用 Gazebo 模拟器可视化的模拟环境提供了一个界面。通过 Gazebo&#xff0c;您可以在真实模拟的物理场景中使用机器人进行测试和实验&#xff0c;并获得高质量的图形。 Gazebo 可在…

鸿蒙4.0核心技术-WebGL开发

场景介绍 WebGL主要帮助开发者在前端开发中完成图形图像的相关处理&#xff0c;比如绘制彩色图形等。 接口说明 表1 WebGL主要接口列表 接口名描述canvas.getContext获取canvas对象上下文。webgl.createBuffer(): WebGLBuffernullwebgl.bindBuffer(target: GLenum, buffer: …

打开VScode时不打开上次使用的文件夹

是不是很烦VScode 打开新的文件夹&#xff0c;每次都打开上次使用过的文件夹&#xff0c;只需在设置里面改一个设置就可以避免了。 Ctrl &#xff0c;打开设置&#xff0c;搜索 window.restoreWindows 通过这种设置就可以让VScode 每次打开新的文件夹而不打开上次的文件夹。

Apache RocketMQ 5.0 腾讯云落地实践

Apache RocketMQ 发展历程回顾 RocketMQ 最早诞生于淘宝的在线电商交易场景&#xff0c;经过了历年双十一大促流量洪峰的打磨&#xff0c;2016年捐献给 Apache 社区&#xff0c;成为 Apache 社区的顶级项目&#xff0c;并在国内外电商&#xff0c;金融&#xff0c;互联网等各行…

ST股票预测模型(机器学习_人工智能)

知己知彼&#xff0c;百战不殆&#xff1b;不知彼而知己&#xff0c;一胜一负&#xff1b;不知彼&#xff0c;不知己&#xff0c;每战必贻。--《孙子兵法》谋攻篇 ST股票 ST股票是指因连续两年净利润为负而被暂停上市的股票&#xff0c;其风险较高&#xff0c;投资者需要谨慎…

OpenCV4工业缺陷检测的六种方法

【文末送书】今天推荐一本机器视觉领域优质书籍 机器视觉 机器视觉是使用各种工业相机&#xff0c;结合传感器跟电气信号实现替代传统人工&#xff0c;完成对象识别、计数、测量、缺陷检测、引导定位与抓取等任务。其中工业品的缺陷检测极大的依赖人工完成&#xff0c;特别是…

微信Windows版-无效的WeChatWin.dll文件,错误码126

更新的微信Windows最新版本&#xff0c;突然有一天打开微信提示“无效的WeChatWin.dll文件 错误码 ErrorCode:126,点击“确定”下载最新版本”。 卸载重新安装跟到windows目录下替换WeChatWin.dll皆无效 该解决方案适用于Windows系统&#xff1a;Windows7、Windows10、Windows…

【HarmonyOS开发】ArkUI中的自定义弹窗

弹窗是一种模态窗口&#xff0c;通常用来展示用户当前需要的或用户必须关注的信息或操作。在弹出框消失之前&#xff0c;用户无法操作其他界面内容。ArkUI 为我们提供了丰富的弹窗功能&#xff0c;弹窗按照功能可以分为以下两类&#xff1a; 确认类&#xff1a;例如警告弹窗 Al…

t-SNE高维数据可视化实例

t-SNE&#xff1a;高维数据分布可视化 实例1&#xff1a;自动生成一个S形状的三维曲线 实例1结果&#xff1a; 实例1完整代码&#xff1a; import matplotlib.pyplot as plt from sklearn import manifold, datasets """对S型曲线数据的降维和可视化"&q…

proxysql读写分离组件部署

一、前言 在mysql一主两从架构的前提下&#xff0c;引入读写分离组件&#xff0c;可以极大的提高mysql性能&#xff0c;proxysql可以在高可用mysql架构发生主从故障时&#xff0c;进行自动的主从读写节点切换&#xff0c;即当mysql其他从节点当选新的主节点时&#xff0c;proxy…

状态管理@Prop

目录 1、父组件State到子组件Prop简单数据类型同步 2、父组件State数组项到子组件Prop简单数据类型同步 3、从父组件中的State类对象属性到Prop简单类型的同步 Prop主要用用于父组件到子组件的数据级联更新&#xff0c;父组件的数据变化会影响到子组件的数据变化&#xff0c…

Arma3/武装突袭3东风战役最后一关游戏无法保存的解决办法

Arma3这个游戏玩进去还是非常有可玩性的&#xff0c;可是在玩过了它本体自带的东风系列战役后&#xff0c;在最精髓的最后一关——game over这个关卡&#xff0c;却有个非常头疼的问题。 逃跑其实是非常简单的&#xff0c;但是想要无伤环游全岛确十分困难&#xff0c;因为这关卡…

Lazada商品详情API在电商中的价值及实时数据获取实践

一、引言 在电商行业&#xff0c;数据是驱动业务增长的关键。Lazada作为东南亚地区知名的电商平台&#xff0c;其商品详情API对于电商行业具有深远的影响。本文将探讨Lazada商品详情API在电商行业中的重要性&#xff0c;并介绍如何实现实时数据获取。 二、Lazada商品详情API的…

秋招总结_就业

2020秋招总结 【前言】 以下内容是写给研二学弟学妹们的秋招总结&#xff0c;研一的师弟师妹们如有需要&#xff0c;也可看看。先说一下我为什么要写这个总结&#xff1a; 1、时代在变化&#xff0c;社会在发展&#xff0c;一届有必要给下一届讲一些经验。 2、我平时和你们…

强大矢量图编辑器 Boxy SVG 激活最新

Boxy SVG for Mac功能介绍 1、干净&#xff0c;直观的UI深受Inkscape&#xff0c;Sketch和Adobe illustrator的启发 2、广泛支持画布上编辑对象几何&#xff0c;转换&#xff0c;绘画和其他属性 3、保存为SVG和SVGZ格式&#xff0c;导出为PNG&#xff0c;JPG&#xff0c;WebP和…

第二证券:耐心等待指数企稳 关注数据要素等板块

其时或是布局商场的时间窗口。2024年美联储或打开降息周期&#xff0c;欧洲降息预期也在增强&#xff0c;全球流动性有望迎来拐点。中心经济工作提及2024年的职业及工业方向&#xff0c;政策有望落地施行&#xff0c;将会为职业配备带来较多的催化要素&#xff0c;是具备配备价…

RTLS 解决现实场景的25种问题

实时定位系统 (RTLS) 可以改变我们理解周围世界的方式。 想想你有多少次停下来寻找某样东西。您可能正在寻找钥匙、电话、鞋子、零食、背包、办公室中的重要文件、停车场中的车辆&#xff0c;或者可能正在寻找街道标志或地标来引导自己。我们大多数人每天都会希望更好地了解我…

【参天引擎】华为参天引擎内核架构源码架构,多线程服务,数据节点管理,多节点间元数据管理

cantian引擎源码结构 ​专栏内容&#xff1a; 参天引擎内核架构 本专栏一起来聊聊参天引擎内核架构&#xff0c;以及如何实现多机的数据库节点的多读多写&#xff0c;与传统主备&#xff0c;MPP的区别&#xff0c;技术难点的分析&#xff0c;数据元数据同步&#xff0c;多主节点…

分享一个好看的vs主题

最近发现了一个很好看的vs主题&#xff08;个人认为挺好看的&#xff09;&#xff0c;想要分享给大家。 主题的名字叫NightOwl&#xff0c;和vscode的主题颜色挺像的。操作方法也十分简单&#xff0c;首先我们先在最上面哪一行找到扩展。 然后点击管理扩展&#xff0c;再搜索栏…
最新文章