C++学习——如何析构派生类

C++——继承关系中的虚函数

  • 析构派生类
  • 纯虚构函数和抽象类

析构派生类

先看一段简单的代码:

#include <iostream>using namespace std;class AA
{
public:AA() {cout << "调用了基类构造" << endl;}virtual void func() {cout << "调用了基类 func()" << endl;}~AA() {cout << "调用了基类析构!" << endl;}};class BB : public AA
{
public:BB() {cout << "调用了子类构造" << endl;}void func() {cout << "调用了子类 func()" << endl;}~BB() {cout << "调用了子类析构!" << endl;}};int main() {// AA aa;BB bb;return 0;
}

编译运行的结果如下:

调用了基类构造
调用了子类构造
调用了子类析构!
调用了基类析构!

修改一下main()函数:

int main() {BB* bb_ptr = new BB;delete bb_ptr;return 0;
}

编译运行的结果与之前一样:

调用了基类构造
调用了子类构造
调用了子类析构!
调用了基类析构!

再修改一下代码,手工调用子类的析构函数:

int main() {BB* bb_ptr = new BB;bb_ptr->~BB();bb_ptr->~BB();bb_ptr->~BB();delete bb_ptr;return 0;
}

编译运行的结果如下:

调用了基类构造
调用了子类构造
调用了子类析构!
调用了基类析构!
调用了子类析构!
调用了基类析构!
调用了子类析构!
调用了基类析构!
调用了子类析构!
调用了基类析构!

可以发现,每次手工调用子类的析构函数,都会调用一次基类的析构函数。
子类的析构函数在执行完成后,会自动调用基类的析构函数。这是C++编译器强制的规定。
继续修改代码,用基类指针指向派生类new出来的对象,代码如下:

int main() {AA* aa_ptr = new BB; // 基类指针指向派生类对象delete aa_ptr; // 基类指针释放派生类对象return 0;
}

编译运行的结果如下:

调用了基类构造
调用了子类构造
调用了基类析构!

用基类指针指向派生类对象是多态的精髓。但用基类指针销毁派生对象时,不能调用派生类析构函数。在应用开发中,一般会把释放资源的代码写在析构函数中。如果析构函数未调用,会有内存泄漏的风险。
但上面这段代码运行的效果没有调用派生类的析构函数,如果需要调用派生类的虚构函数,则把基类的析构函数声明为虚构函数即可,代码如下:

#include <iostream>using namespace std;class AA
{
public:AA() {cout << "调用了基类构造" << endl;}virtual void func() {cout << "调用了基类 func()" << endl;}// 基类的析构函数是虚函数virtual ~AA() {cout << "调用了基类析构!" << endl;}};class BB : public AA
{
public:BB() {cout << "调用了子类构造" << endl;}void func() {cout << "调用了子类 func()" << endl;}~BB() {cout << "调用了子类析构!" << endl;}};int main() {AA* aa_ptr = new BB; // 基类指针指向派生类对象delete aa_ptr; // 基类指针释放派生类对象return 0;
}

编译运行的结果如下:

调用了基类构造
调用了子类构造
调用了子类析构!
调用了基类析构!

正常情况下,释放堆区内存后,会把指针指向空,防止操作野指针。delete空指针是安全的,delete野指针会导致程序的崩溃。

delete ptr;
ptr = nullptr;

如果析构函数被调用了多次,如果没有ptr = nullptr;,那么delete ptr;操作的就是野指针。
所以,在析构函数中,释放堆区内存后,也应该把指针指向空。
对于基类,即使不需要析构函数,也应该提供一个空虚析构函数。目的是为了让派生类有机会可以重写析构函数。
下面的代码就演示了,基类没有析构函数,派生类有析构函数的情况:

#include <iostream>using namespace std;class AA
{
public:AA() {cout << "调用了基类构造" << endl;}virtual void func() {cout << "调用了基类 func()" << endl;}// 基类缺少析构函数// virtual ~AA() {cout << "调用了基类析构!" << endl;} };class BB : public AA
{
public:BB() {cout << "调用了子类构造" << endl;}void func() {cout << "调用了子类 func()" << endl;}~BB() {cout << "调用了子类析构!" << endl;}};int main() {AA* aa_ptr = new BB; // 基类指针指向派生类对象delete aa_ptr; // 基类指针释放派生类对象return 0;
}

编译运行结果如下:

调用了基类构造
调用了子类构造

可以发现,销毁派生类对象时,没有调用析构函数,因为基类指针不能调用派生类的成员函数。若想调用派生类的构造函数,在基类中添加一个空的虚构函数即可。代码如下:

class AA
{
public:AA() {cout << "调用了基类构造" << endl;}virtual void func() {cout << "调用了基类 func()" << endl;}// virtual ~AA() {cout << "调用了基类析构!" << endl;}virtual ~AA() {} // 空的析构函数
};

编译运行的结果如下:

调用了基类构造
调用了子类构造
调用了子类析构!

纯虚构函数和抽象类

纯虚函数是一种特殊的虚函数,在某些情况下,基类中不能对虚函数给出有意义的实现,把它声明为纯虚函数。
纯虚函数只有函数名、参数和返回值类型,没有函数体,具体实现留给该派生类去做。
语法:virtual 返回值类型 函数名 (参数列表)=0;
纯虚函数基类中为派生类保留一个函数的名字,以便派生类它进行重定义。如果在基类中没有保留函数名字,则无法支持多态性。
含有纯虚函数的类被称为抽象类,不能实例化对象,可以创建指针和引用。
派生类必须重定义抽象类中的纯虚函数,否则也属于抽象类。
比较好理解,直接看代码:

#include <iostream>using namespace std;class AA // 含有纯虚函数的类——抽象类
{
public:AA() {cout << "调用了基类构造" << endl;}virtual void func() = 0; // 纯虚函数virtual ~AA() {cout << "调用了基类析构!" << endl;}
};class BB : public AA // 派生类
{
public:BB() {cout << "调用了子类构造" << endl;}void func() {cout << "调用了子类 func()" << endl;}~BB() {cout << "调用了子类析构!" << endl;}
};int main() {BB bb; // 创建派生类对象AA* aa_ptr = &bb; // 创建抽象类指针,指向子类对象。aa_ptr->func(); // 调用子类 func()// AA& aa_ref = bb; // 创建抽象类引用,指向子类对象。// aa_ref.func(); // 调用子类 func()return 0;
}

编译运行的效果如下:

调用了基类构造
调用了子类构造
调用了子类 func()
调用了子类析构!
调用了基类析构!

基类中的纯虚析构函数也需要实现。

class AA // 抽象类
{
public:AA() {cout << "调用了基类构造" << endl;}virtual void func() = 0;virtual ~AA() = 0;
};

这段代码编译会通过,但是链接过程中会报错。原因是:当派生类对象被销毁时,会调用派生类的析构函数,接着自动调用基类的析构函数。现在基类析构函数没有代码实现了,所以报错了。这种错误是链接阶段的,不是编译阶段的。
既然纯虚析构函数一定要有代码实现,那它还有啥意义?
有时候,想使一个类成为抽象类,但刚好又没有任何纯虚函数,怎么办?
方法很简单:在想要成为抽象类的类中声明一个纯虚析构函数。
纯虚析构函数的定义要在类的外部

class AA // 抽象类
{
public:AA() {cout << "调用了基类构造" << endl;}virtual void func() = 0;virtual ~AA() = 0; // 纯虚析构函数声明
};AA::~AA() {cout << "调用了基类析构!" << endl;}; // 纯虚析构函数定义

感谢浏览,一起学习!

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

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

相关文章

CentOS7.X 安装RustDesk自建服务器实现远程桌面控制

参照文章CentOS安装RustDesk自建服务器中间总有几个位置出错&#xff0c;经实践做个记录防止遗忘 一 环境&工具准备 1.1 阿里云轻量服务器、Centos7系统、目前最高1.1.11版本rustdesk-server-linux-amd64.zip 1.2 阿里云轻量服务器–安全组–开放端口&#xff1a;TCP(21…

工具篇:IDEA VFS 损害启动报错 com.intellij.util.io.CorruptedException 处理

文章目录 前言一、 idea 的 VFS是什么&#xff1f;二、解决方式&#xff1a;2.1 退出Idea 然后重新打开&#xff1a;2.2 手动清除Idea 缓存&#xff0c;让Idea 重新建立缓存&#xff1a;2.2.1 打开 Invalidate Caches / Restart 对话框:2.2.2 勾选要清除的缓存&#xff1a; 总结…

2.linux中调度kettle

一.准备转换&#xff0c;等会在linux中用 1.添加excel输入组件&#xff0c;并添加对应的文件 2.添加列拆分为多行组件 3.添加文本文件输出组件 4.保存转换 二.linux安装java 1.把jdk-8u144-linux-x64.tar.gz上传到linux的/lx目录下 2. 解压jdk包&#xff0c;然后配置环境变量…

第四节、电机定角度转动【51单片机-TB6600驱动器-步进电机教程】

摘要&#xff1a;本节介绍用电机转动角度计算步骤&#xff0c;从而控制步进电机转角 一、 计算过程 1.1 驱动器接收一个脉冲后&#xff0c;步进电机转动一步&#xff0c;根据驱动器设置的细分值 计算一个脉冲对应电机转动的角度step_x s t e p x s t e p X … … ① step_{x…

如何终身使用 100% 免费的服务器

作为开发人员,我们需要在云服务上运行和托管后端。有许多 BaaS(后端即服务)可用,但它们有一些限制。 如果我说我已经免费使用基于 Linux 的服务器超过 4-5 年了,那会怎样?是的,你没听错。我正在使用这台安装了 Ubuntu 20、24 GB RAM、4 个 CPU 和 200 GB 存储空间的 Lin…

【计算机组成原理】期末复习题库

5&#xff0e;主存储器和CPU之间增加cache的目的是 。 A&#xff0e;解决CPU和主存之间的速度匹配问题 B&#xff0e;扩大主存储器的容量 C&#xff0e;扩大CPU中通用寄存器的数量 D&#xff0e;既扩大主存容量又扩大CPU中通用寄存器的数量 在计算机系统中&#xff0c;CPU的速…

SAP中Smartforms 翻译越南语

点击打印预览 打印预览中确实是越南语 转出成PDF 成了乱码 SPAD中查询LP01其实是简体中文 换成LP02试试 显示看上去正常的 SPAD中的LP02 SU3可以设置自己的默认打印参数 查查Smartforms中的字体样式 是宋体&#xff0c;看上去不用为了越南文刻意改字体样式成TIMES 看这篇文章…

26.删除有序数组中的重复项 python

删除有序数组中的重复项 题目题目描述示例 1&#xff1a;示例 2&#xff1a;提示&#xff1a;题目链接 题解解题思路python实现代码解释提交结果 题目 题目描述 给你一个 非严格递增排列 的数组 nums &#xff0c;请你 原地 删除重复出现的元素&#xff0c;使每个元素 只出现…

R语言 | 峰峦图 / 山脊图

目的&#xff1a;为展示不同数据分布的差异。 1. ggplot2 实现 # 准备数据 datmtcars[, c("mpg", "cyl")] colnames(dat)c("value", "type") head(dat) # value type #Mazda RX4 21.0 6 #Mazda RX4 Wag …

四川创新志成健康管理有限公司

四川创新志成健康管理有限公司 成都市青羊区广富路168号 公司简介 四川创新志成健康管理有限公司成立于2021年&#xff0c;公司专注体外诊断领域&#xff0c;致力为医学实验室、生产厂家、 经销商提供专业的学术、技术增值服务&#xff0c;涵盖免疫、生化、输血等检测领域&a…

系统级 I/O

Unix I/O **了解 Unix I/O 将帮助你理解其他的系统概念。**I/O 是系统操作不可或缺的一部分。我们经常遇到 I/O 和其他系统概念之间的循环依赖。例如&#xff0c;I/O 在进程的创建和执行中扮演着关键的角色。反过来&#xff0c;进程创建又在不同进程间的文件共享中扮演着关键角…

Elasticsearch:使用阿里 infererence API 及 semantic text 进行向量搜索

在之前的文章 “Elasticsearch 开放推理 API 新增阿里云 AI 搜索支持”&#xff0c;它详细描述了如何使用 Elastic inference API 来针对阿里的密集向量模型&#xff0c;稀疏向量模型&#xff0c; 重新排名及 completion 进行展示。在那篇文章里&#xff0c;它使用了很多的英文…

基于公网的无线全双工内部通话系统在演出行业可以用吗?

文旅名城再出发&#xff0c;更待“烟花”绽繁花 2024年4月将开业的扬州首个大型沉浸式剧场-《运河密城》 以运河为原点 追随河的记忆 从春秋时代的吴王夫差 到贯通南北的大运河成形 穿梭时空 探索扬州的前世今生 「运河第一锹」古运河旁 有一处新地标正在悄然兴起 如…

POSTGRESQL跟ORACLE语法区别和相同之处

跟ORACLE语法区别之处 1. Update和delete语法区别 Pg 和MySQL Update和delete的时候表名不能加别名 2. 插入数字类型不一样 ORACLE 对number类型的数据可以用’’ 字符串标记插入&#xff0c;但是PG不行&#xff0c;必须要进行正确的数据类型 3. SEQ使用不同 ORACEL的SEQ…

C++编程物联网:舵机VS步进电机

舵机和步进电机都是常见的电机类型,它们在自动化和机器人控制中有着不同的应用场景。两者的主要区别在于控制方式、运动精度、适用范围等方面。下面详细介绍它们的作用、应用场景和主要区别。 1. 舵机(Servo Motor) 工作原理 舵机是一种具有反馈控制的电动机,通常由电动…

鸿翼参与撰写档案数据管理与长期保存策略基于数字中国战略的研究

​编者按&#xff1a;近日&#xff0c;由中国财富出版社有限公司出版的《档案数据管理与长期保存策略——基于数字中国战略的研究》正式发行&#xff0c;上海鸿翼软件技术股份有限公司董事长兼CEO龙凌云作为核心作者参与主要编写工作。 本书是在国家档案局立项科研项目“数字档…

机器学习中的图匹配问题—基础学习

机器学习中的图匹配问题 结合导师所给的方向&#xff0c;能否将实例之间的点匹配问题转换为点到实例之间的匹配问题来进行求解呢&#xff1f;这里结合师姐推荐的讲座首先对图匹配的这个方向来进行简单的了解和接触。 图匹配问题概述 图匹配就是&#xff1a;不仅考虑点之间的配…

2024.11.29——[HCTF 2018]WarmUp 1

拿到题&#xff0c;发现是一张图&#xff0c;查看源代码发现了被注释掉的提示 <!-- source.php--> step 1 在url传参看看这个文件&#xff0c;发现了这道题的源码 step 2 开始审计代码&#xff0c;分析关键函数 //mb_strpos($haystack,$needle,$offset,$encoding):int|…

gRPC 快速入门 — SpringBoot 实现(1)

目录 一、什么是 RPC 框架 &#xff1f; 二、什么是 gRPC 框架 &#xff1f; 三、传统 RPC 与 gRPC 对比 四、gRPC 的优势和适用场景 五、gRPC 在分布式系统中应用场景 六、什么是 Protocol Buffers&#xff08;ProtoBuf&#xff09;&#xff1f; 特点 使用场景 简单的…

工具篇--GitHub Desktop 使用

文章目录 前言一、GitHub Desktop 的使用&#xff1a;1.1 通过官网下载GitHub Desktop和安装&#xff1a;1.2 安装和使用&#xff1a;1.2.1 填充自己的标识&#xff1a;1.2.3 克隆项目&#xff1a;1.2.4 git 常用忽略项配置&#xff1a; 二、代码的更新和提交&#xff1a;2.1 代…