C++|多态性与虚函数(2)|虚析构函数|重载函数|纯虚函数|抽象类

news2024/11/16 23:58:57

前言

看这篇之前,可以先看多态性与虚函数(1)⬇️

C++|多态性与虚函数(1)功能绑定|向上转换类型|虚函数-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/weixin_74197067/article/details/138861418?spm=1001.2014.3001.5501这篇文章会接着上一篇来记录

虚析构函数

构造函数不能是虚函数,但析构函数可以是虚函数,同样的在析构函数前面加上virtual就 称该析构函数为虚析构函数,语法如下:

virtual ~类名() 

{

        函数体

}

#include<iostream>
using namespace std;

class point
{
public:
	point():x(0),y(0){}
	point(double a, double b) :x(a), y(b) {}
	virtual ~point()
	{
		cout << "~point" << endl;
	}

	double x;
	double y;

};

class rectangle :public point
{
public:
	rectangle():point(0,0),x1(0),y1(0){}
	rectangle(double a, double b, double c, double d) :point(a, b), x1(c), y1(d) {}
	~rectangle()
	{
		cout << "~rectangle" << endl;
	}
private:
	double x1;
	double y1;
};

int main()
{
	point* p = nullptr;
	p = new rectangle;
	delete p;
	return 0;
}

就会先调用rectangle 的析构函数,在调用point的析构函数。如果在基类的析构函数中不加virtual,那么运行结果只会调用point的析构函数

如果将基类的虚构函数写为虚函数,那么后面所有继承该基类的派生类的所有析构函数自动成为虚函数,即使名字不相同。

当基类的析构函数为虚函数时,无论指针指向的是同一类族那个类的对象,程序都会动态关联,调用相应的析构函数,对该对象所涉及的额外内存空间进行清理工作。

最好把基类的析构函数声明为虚函数,这使得后面派生类所有的析构函数均为虚函数,这样如果调用delete运算符准备删除一个对象,而delete运算符的操作对象用了派生类对象的基类指针,系统就会先调用派生类的析构函数,在调用基类的析构函数,这样整个派生类的对象被完全释放。

虚函数与重载函数比较

函数重载处理的是同一层次的函数重名问题,

而虚函数处理的是同一类族中不同派生层次 上的函数重名问题

函数重载时,参数个数和参数类别可以不一样

虚函数必须保证同名,同参数,同返回值

函数重载可以是成员函数也可以普通函数

虚函数只能是成员函数

函数重载的调用是以所传递参数序列的差别作为调用不同函数的依据

虚函数是根据对象的不同去调用不同类的虚函数

函数重载在编译时表现出多态性

虚函数在运行时表现多态性

纯虚函数

有时,基类往往表示一种抽象的概念,并不与其他事物相连,例如,封闭图形只是一个概念,但他可以派生出,三角形、四边形、五边形...

纯虚函数与一般虚函数定义格式基本相同,只是在后面多了”=0“部分表示没有函数体部分,纯虚函数的函数体由派生类给出

纯虚函数的定义形式如下:

class 类名

{

        ...

        virtual 函数类型 函数名(参数表)=0;

        ...

}

下面演示一下:基类是封闭图形,派生类有三角形,四边形。

代码如下所示:

#include<iostream>
using namespace std;
class shape
{
public:
	virtual void show() = 0;
};

class triangle :public shape
{
public:

	triangle(double a,double b,double c):a(a),b(b),c(c){}
	void show()
	{
		cout << "triangle" << endl;
	}
private:
	double a;
	double b;
	double c;

};

class quadrangle :public shape
{
public:
	quadrangle(double a,double b,double c,double d):a(a),b(b),c(c),d(d){}
	void show()
	{
		cout << "quardrangle" << endl;
	}
private:
	double a;
	double b;
	double c;
	double d;
};

int main()
{
	shape *s=nullptr;
	triangle t(1, 2, 3);
	quadrangle q(1, 2, 3, 4);
	s = &t;
	s->show();

	s = &q;
	s->show();
	return 0;
}

抽象类

如果一个类至少有一个纯虚函数,那么就称该类为抽象类,因此上面那个shape类就是抽象类。

对于抽象类的使用规则:

1.因为抽象类 中至少包含一个没有定义功能的纯虚函数,抽象类只能作为其他类的基类的来使用,不能建立抽象类对象,只能为派生类提供一个接口规范,其功能由派生类给出。

2.不允许从具体类中派生出抽象类

3.抽象类不能用作参数类型,函数返回值类型或者显示转换类型

4.可以声明抽象类类型的指针和引用,此指针可以指向它的派生类对象进而实现动态多态性

5.如果这个派生类没有重写虚函数,那么这个 派生类就是简单继承这个纯虚函数,那么这个派生类就是一个抽象类

6.在抽象类中可以定义普通成员函数或者虚函数

注意函数体为空的虚函数不是纯虚函数,函数体为空不代表没有函数体,纯虚函数没有函数体

override

如果一个成员函数是虚函数,那么在后续派生类里的同名函数都是虚函数,无须再用virtual修饰。这样就会出现以下问题:

1.这样在函数很多的情况下,就难以分辨哪些函数是派生类特有的成员函数,哪些函数继承自基类;

2.还有可能派生类可能无意使用了一个同名但函数原型不同的函数”覆盖“了基类的虚函数

#include<iostream>
using namespace std;
class base
{
public:
	virtual void f() = 0;//纯虚函数
	virtual void g()const//虚函数
	{
		cout << "base:g()" << endl;
	}
	 void h()//一般成员函数
	{
		cout << "base:h()" << endl;
	}
};

class derived :public base
{
public:
	 void f()//纯虚函数重写
	{
		cout << "derived:f()" << endl;
	}
	 void g()//不是虚函数重新,遗漏了const修饰
	{
		cout << "derived:g()" << endl;
	}
	 void h()//一般成员函数
	{
		cout << "derived:h()" << endl;
	}
};


int main()
{
	base* b = nullptr;
	derived c;
	b = &c;
	b->f();
	b->g();
	b->h();
	return 0;
}

C++11增加了一个特殊的标识符override,它可以显示地标记虚函数重写 ,明确代码编写者的意图:派生类成员函数名后面有override修饰,那么这个函数就一定是虚函数,而且函数原型也必须与基类声明一致,否则编译错误。

void f()override//纯虚函数重写
{
	cout << "derived:f()" << endl;
}

final

final可以在类名后面使用,显式的禁止类被继承,不能再有派生类

final可以在虚函数后面使用,显式的禁止该函数在派生类中再次被重写

例如

#include<iostream>
using namespace std;

class base
{
public:
	virtual void f() = 0;//纯虚函数
	virtual void g()//虚函数
	{
		cout << "base:g()" << endl;
	}
};

class derived :public base
{
public:
	void f()override final//纯虚函数重写,并不允许被重写
	{
		cout << "derived:f()" << endl;
	}
	void g()override//虚函数重写
	{
		cout << "derived:g()" << endl;
	}

};

class last final :public derived//表示last不允许被继承,last也不会再有派生类
{
	/*void f()override //纯虚函数重写,并不允许被重写
	{cout << "last:f()" << endl;}*/
	void g()//虚函数重写
	{
		cout << "last:g()" << endl;//可以重写
	}
};

 

last类在类名后使用了final,使它变成了类体系的终点

在derived类的函数f后使用了final,表示在后面的派生类中禁止重写该虚函数。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Kasawaki川崎机器人故障维修

在当今的自动化工业领域&#xff0c;川崎工业机器人以其卓越的性能和可靠的工作效率赢得了广泛的赞誉。作为机器人的核心组成部分&#xff0c;伺服电机的作用至关重要。然而&#xff0c;就像所有机械设备一样&#xff0c;也可能会遭遇电机磨损或故障&#xff0c;需要适时的川崎…

二叉树——初解

二叉树 树树的概念树的性质 二叉树二叉树的概念二叉树的性质二叉树的实现方式数组构建左孩子右兄弟法构建指针构建 树 树的概念 在计算机科学中&#xff0c;树&#xff08;Tree&#xff09;是一种重要的非线性数据结构&#xff0c;它由若干节点&#xff08;Node&#xff09;组…

iPhone15销量不佳,新产品没希望,苹果开始寻找库克接班人

美国媒体开始谈论谁将成为苹果的新CEO&#xff0c;这意味着苹果董事会开始为库克寻找接班人了&#xff0c;导致如此结果&#xff0c;可能在于iPhone15的表现实在太差了&#xff0c;而库克力推的vision Pro等新产品又没有为苹果打开局面所致。 如今的苹果倒是与1980年代乔布斯离…

壹资源知识付费系统源码-小程序端+pc端

最新整理优化&#xff0c;含微信小程序和pc网页。内置几款主题&#xff0c;并且可以自己更改主题样式&#xff0c;各区块颜色&#xff0c;文字按钮等。 适用于知识付费类资源类行业。如&#xff1a;项目类&#xff0c;小吃技术类&#xff0c;图书类&#xff0c;考研资料类&…

Nodejs 第七十一章(libuv)

libuv 在Node.js中&#xff0c;libuv是作为其事件循环和异步I/O的核心组件而存在的。Node.js是构建在libuv之上的&#xff0c;它利用libuv来处理底层的异步操作&#xff0c;如文件I/O、网络通信和定时器等。 libuv在Node.js中扮演了以下几个重要角色&#xff1a; 事件循环&a…

同为科技详解智能PDU所应用的通信协议与接口

现如今&#xff0c;信息服务、AI人工智能的飞速发展与增长&#xff0c;全球正经历信息数据的爆炸。不仅数据量以惊人的速度增长&#xff0c;而且全球社会各行业对数据的依赖的程度也在日益增加。这些趋势使数据中心在全球都享有关键基础架构的地位。假设某个数据中心发生严重的…

写一个类ChatGPT应用,前后端数据交互有哪几种

❝ 对世界的态度&#xff0c;本质都是对自己的态度 ❞ 大家好&#xff0c;我是「柒八九」。一个「专注于前端开发技术/Rust及AI应用知识分享」的Coder 前言 最近&#xff0c;公司有一个AI项目&#xff0c;要做一个文档问答的AI产品。前端部分呢&#xff0c;还是「友好借鉴」Cha…

经典神经网络(8)GAN、CGAN、DCGAN、LSGAN及其在MNIST数据集上的应用

经典神经网络(8)GAN、CGAN、DCGAN、LSGAN及其在MNIST数据集上的应用 1 GAN的简述及其在MNIST数据集上的应用 GAN模型主导了生成式建模的前一个时代&#xff0c;但由于训练过程中的不稳定性&#xff0c;对GAN进行扩展需要仔细调整网络结构和训练考虑&#xff0c;因此GANs虽然在…

【js逆向】易车网JS逆向案例实战手把手教学(附完整代码)

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全…

一表捋清网络安全等级保护测评要求

三级网络安全等级保护测评指标&#xff1a; 对于中小企事业单位来说&#xff0c;网络安全建设是一个复杂且投入较高的过程&#xff0c;因此他们更倾向于寻找一种“省心省力”的等保建设方案&#xff0c;以及一种能够持续有效且具有较高性价比的网络安全建设投入方式。 此时&…

合合信息:TextIn文档解析技术与高精度文本向量化模型再加速

文章目录 前言现有大模型文档解析问题表格无法解析无法按照阅读顺序解析文档编码错误 诉求文档解析技术技术难点技术架构关键技术回根溯源 文本向量化模型结语 前言 随着人工智能技术的持续演进&#xff0c;大语言模型在我们日常生活中正逐渐占据举足轻重的地位。大模型语言通…

通过java将数据导出为PDF,包扣合并单元格操作

最近项目中需要将查询出来的表格数据以PDF形式导出&#xff0c;并且表格的形式包含横向行与纵向列的单元格合并操作&#xff0c;导出的最终效果如图所示&#xff1a; 首先引入操作依赖 <!--导出pdf所需包--><dependency><groupId>com.itextpdf</groupId&…

Linux 生态与工具

各位大佬好 &#xff0c;这里是阿川的博客 &#xff0c; 祝您变得更强 个人主页&#xff1a;在线OJ的阿川 大佬的支持和鼓励&#xff0c;将是我成长路上最大的动力 阿川水平有限&#xff0c;如有错误&#xff0c;欢迎大佬指正 目录 Linux生态简介:Linux工具lrzsz&#xff…

适用于 Windows 8/10/11 的 10 大 PC 迁移工具:电脑克隆迁移软件

当您发现自己拥有一台新的 PC 或笔记本电脑时&#xff0c;PC 迁移变得至关重要。将数据从旧计算机传输到新计算机的过程似乎令人生畏&#xff0c;尤其是如果您是第一次这样做。迁移过程中数据丢失的潜在风险加剧了焦虑。为确保文件和系统设置的无缝无忧传输&#xff0c;使用专为…

探索设计模式的魅力:机器学习赋能,引领“去中心化”模式新纪元

​&#x1f308; 个人主页&#xff1a;danci_ &#x1f525; 系列专栏&#xff1a;《设计模式》 &#x1f4aa;&#x1f3fb; 制定明确可量化的目标&#xff0c;坚持默默的做事。 探索设计模式的魅力&#xff1a;机器学习赋能&#xff0c;引领“去中心化”模式新纪元 ✨欢迎加入…

3月份太阳镜行业线上市场销售数据分析

在消费者行为方面&#xff0c;太阳镜不仅仅是视力保护工具&#xff0c;更逐渐成为一种时尚单品。随着人们对健康和美容重视程度的提高&#xff0c;太阳镜作为体现个人风格的单品&#xff0c;其市场需求得到了进一步的推动。此外&#xff0c;全球旅行和旅游业的恢复&#xff0c;…

被暗示离职?教你优雅反击

在职场中&#xff0c;面对公司暗示离职的情况&#xff0c;应届毕业生可能会感到困惑与无助。但是&#xff0c;保持冷静和据理力争是保护自己权益的重要途径。以下是如何应对此类情况的一些建议。 当你感觉到被暗示离职时&#xff0c;首要的策略就是与上司进行有效沟通。安排一个…

halo博客--解决恶意刷评论的问题

原文网址&#xff1a;halo博客--解决恶意刷评论的问题_IT利刃出鞘的博客-CSDN博客 简介 本文介绍halo博客如何通过设置评论次数来解决恶意刷评论的问题。 评论功能要设置频率的限制&#xff0c;否则可能被人一直刷评论&#xff0c;然后数据库存的垃圾评论越来越多&#xff0…

数据结构——队列(链表实现)

一、队列的特点 先进先出 二、队列的代码 typedef int QDataType;// 链式结构&#xff1a;表示队列 typedef struct QListNode {struct QListNode* next;QDataType data; }QNode;// 队列的结构 typedef struct Queue {QNode* front; //指向队列的第一个结点QNode* rear;//指…

刷代码随想录有感(66):回溯算法——组合问题的优化(剪枝)

代码&#xff1a;将for循环中i的搜索范围进行缩小&#xff0c;免去多余的不可能符合条件的操作。 for(int i start; i < n-(k-tmp.size())1;i) 实质是剪枝&#xff0c;拿n4,k4作比较&#xff1a; 显然结果只可能是[1,2,3,4]&#xff0c;选取顺序只可能是1-2-3-4&#xff…