析构函数详解

目录

  • 析构函数
    • 概念
    • 特性
    • 对象的销毁顺序

感谢各位大佬对我的支持,如果我的文章对你有用,欢迎点击以下链接
🐒🐒🐒 个人主页
🥸🥸🥸 C语言
🐿️🐿️🐿️ C语言例题
🐣🐣🐣 python
🐓🐓🐓 数据结构C语言
🐔🐔🐔 C++
🐿️🐿️🐿️ 文章链接目录

析构函数

概念

通过上一篇文章我们知道一个对象是怎么来的,那一个对象又是怎么没呢的?

析构函数:与构造函数功能相反,析构函数不是完成对对象本身的销毁,局部对象销毁工作是由
编译器完成的。

而对象在销毁时会自动调用析构函数,完成对象中资源的清理工作(有点像数据结构中的Destroy函数,作用就是清理链表 树 或堆上的空间清理,如果不清理会出现内存泄漏的情况)

注意对象空间的开辟和销毁不需要我们去解决,这些都是由系统去完成的(全局 对象 静态都是系统自己去解决),而像堆上的空间就需要我们去完成了,比如malloc开辟空间的时候需要我们去完成,在最后释放的时候也是需要我们自己去free掉空间

特性

析构函数是特殊的成员函数,其特征如下:

  1. 析构函数名是在类名前加上字符 ~
  2. 无参数无返回值类型。
    具体结构如下:
class Date
{~Date(){;}
};

类名前的~在C语言中表示按位与取反,这里的取反有完全相反的意思,所以 ~放在析构函数这里就是想说明析构函数的作用和构造函数是完全不同的

特别注意析构函数是没有参数的,而构造函数是有参数的,因为构造函数要构造,传参可以初始化,而析构函数完全没必要传参,所以就没有参数

  1. 一个类只能有一个析构函数。若未显式定义,系统会自动生成默认的析构函数。注意:析构函数不能重载
class Date
{
public:Date(){_year = 1;}~Date(){cout << "~Date()" << endl;}
private:int _year;
};
int main()
{Date d1;return 0;
}

在这里插入图片描述

  1. 对象生命周期结束时,C++编译系统系统自动调用析构函数。

  2. 关于编译器自动生成的析构函数,是否会完成一些事情呢?下面的程序我们会看到,编译器生成的默认析构函数,对自定类型成员调用它的析构函数。

  3. 如果类中没有申请资源时,析构函数可以不写,直接使用编译器生成的默认析构函数
    比如Date类;有资源申请时,一定要写,否则会造成资源泄漏,比如Stack类。

由于析构函数和构造函数都是特殊的类,所以都是有this指针的

class Date
{
public:Date(){_year = 1;}~Date(){cout << this << endl;cout << "~Date()" << endl;}void Print(){cout << this << endl;cout << "Print()" << endl;}
private:int _year;
};
void func()
{Date d2;
}
int main()
{func();Date d1;d1.Print();return 0;
}

通过调试我们可以看到,d1和d2的地址以this指针的方式传给函数,d1和d2在生命周期结束时会调用析构函数,而析构函数里面是打印this指针
在这里插入图片描述
我们来看看下面的代码来具体理解析构函数

typedef int DataType;
class Stack
{
public:Stack(size_t capacity = 3){_array = (DataType*)malloc(sizeof(DataType) * capacity);if (NULL == _array){perror("malloc申请空间失败");return;}_capacity = capacity;_size = 0;}void Push(DataType data){// CheckCapacity();扩容_array[_size] = data;_size++;}//~Stack()//{//	if (_array)//	{//		free(_array);//		_array = NULL;//		_capacity = 0;//		_size = 0;//	}//}
private:DataType* _array;int _capacity;int _size;
};
int main()
{Stack s;s.Push(1);s.Push(2);
}

在C语言中当没有调用Destroy函数会发生内存泄漏,具体过程如下

main函数会在栈上开辟一块空间,这块空间中也包含Stack s的指针DataType* _array(只是这个指针在mian函数开辟的空间里)
在这里插入图片描述
DataType* _array中_arrray的作用是保存Stack s开辟空间的地址
在main函数执行完后,会将main函数在栈上开辟的空间都销毁,其中就包括了指针_array
由于_array是保存着Stack s开辟空间的地址,最终会因为指针_array被销毁,导致找不到Stack s开辟出的空间

在这里插入图片描述
所以没调用Destroy函数发生的后果是很严重的,并且我们经常会忘记调用Destroy函数,为了解决这个问题才有了析构函数,因为析构函数自动调用,并且编译器可以自动生成析构函数,这对我们来说是非常方便的

但是需要注意的是默认生成的析构函数和默认生成的构造函数类似,对内置类型不做处理,自定义类型的成员会去调用他的析构函数

对象的销毁顺序

生命周期对于现在学到的来说有两种,一种是局部(存在一些函数中,因为调用函数会开辟栈帧,所以函数结束后栈帧也会被销毁,函数中的局部变量也就销毁了),另一种是静态或者全局的(存在静态区里,在mian函数结束后就会销毁)

而对象生命周期结束时,C++编译系统系统自动调用析构函数,那如果有多个对象生命周期同时结束,系统会优先给谁调用析构函数

class Date
{
public:Date(int year){_year = year;
}~Date(){cout << "~Date()" << _year<<endl;}
private:int _year;int _month;int _day;
};
int main()
{Date d1(1);Date d2(2);
}

这段代码中我们只定义了一个成员变量_year,其他的_month以及_day都只是声明,不占用内存空间,mian函数中Date d1(1),Date d2(2)是对_year进行初始化,在函数结束后两个对象的生命周期都会结束,而销毁的顺序如图
在这里插入图片描述
这个调用的顺序像栈中的后进先出,Date d1先入栈,所以最后调用析构函数,事实上对象确实存储在栈上的,因为类其实是一个函数,在函数调用时会建立栈帧,所以空间存储在栈上

class Date
{
public:Date(int year){_year = year;
}~Date(){cout << "~Date()" << _year<<endl;}
private:int _year;int _month;int _day;
};
int main()
{Date d1(1);Date d2(2);static Date d3(3);
}

那如果让Date d3加上一个static去修饰结果会怎么样
在这里插入图片描述
加上static修饰后Date d3的存储区域就发生变化了,d3存储在一个单独的静态区中,虽然d3是一个局部变量,但是他的生命周期在经过static修饰后变成全局,所以d3会在main函数结束后销毁,而在main函数结束前会将里面的d1和d2等局部变量先销毁掉,所以d3排在最后

我们再来看看下面的代码,这段代码中定义了一个函数func,将类的对象定义在函数中,其中d4是被static修饰的,而d3没有被修饰,然后在main函数中调用func

class Date
{
public:Date(int year){_year = year;
}~Date(){cout << "~Date()" << _year<<endl;}
private:int _year;int _month;int _day;
};
void func()
{Date d3(3);static Date d4(4);
}
int main()
{Date d1(1);Date d2(2);func();
}

销毁顺序是3 2 1 4,具体原因还是因为func也是一个函数,空间开辟在栈上的,满足后进先出原则,所以先销毁对象d3(对于d3为什么是最先销毁,可能是因为他在函数func中,算是一个局部中的局部吧),然后又是d2 d1,d4因为被static修饰,所以最后销毁
在这里插入图片描述
我们再在main函数外定义一个对象d5又会怎么样

class Date
{
public:Date(int year){_year = year;
}~Date(){cout << "~Date()" << _year<<endl;}
private:int _year;int _month;int _day;
};
void func()
{Date d3(3);static Date d4(4);
}
Date d5(5);
int main()
{Date d1(1);Date d2(2);func();
}

结果是3 2 1 4 5
在这里插入图片描述
d5虽然没有被static修饰,但是他定义在main函数外的,所以他自己就是一个全局变量,但是这里的全局变量有两个,一个是d4,一个是d5,他们的销毁顺序是否也和自己的位置有关呢?

我们将d5的定义移到func函数上边,发现没有变化,所以推测可能是因为d4是在func函数中,所以相对于d5来讲,d4的声明周期是局部的
在这里插入图片描述

class Date
{
public:Date(int year){_year = year;
}~Date(){cout << "~Date()" << _year<<endl;}
private:int _year;int _month;int _day;
};
static Date d6(6);
Date d5(5);
void func()
{Date d3(3);static Date d4(4);
}
int main()
{func();Date d1(1);Date d2(2);

我们再在main函数外定义一个d6,用static修饰他的顺序又会怎么样
在这里插入图片描述
当d6和d5交换顺序后,发现销毁的顺序变化了,所以我们得出结论,全局的销毁顺序和局部的销毁顺序也是一样的,当d5后入栈时,d5就先销毁,而static修饰全局变量d6,并不会改变d6的销毁顺序
在这里插入图片描述
在这里插入图片描述
如果在多个函数func中定义类的顺序会怎么样

class Date
{
public:Date(int year){_year = year;
}~Date(){cout << "~Date()" << _year<<endl;}
private:int _year;int _month;int _day;
};
static Date d6(6);
Date d5(5);
void func2()
{Date d7(7);static Date d8(8);
}
void func1()
{Date d3(3);static Date d4(4);
}int main()
{Date d1(1);Date d2(2);func1();func2();
}

在这里插入图片描述
这里说下我的想法,func1和func2因为都是在栈上开的空间,所以他们的销毁的顺序满足后进先出,具体判断谁先销毁的方法就是看谁最先被调用,也就是在main函数中去看func1是否比func2先调用,如果先比func2调用,那就说明func1先开辟空间,所以func1要比func2后销毁
在这里插入图片描述
最终顺序总结如下
局部对象(后定义先析构)->局部静态->全局对象(后定义先析构)

类中没有显示定义析构函数,系统则会自动生成默认的析构函数,那这个析构函数是否和构造函数一样基本上什么事都不做呢?
由于自定义类型的尽头是内置类型,对应类而言如果类中没有申请资源时,析构函数可以不写(因为不写不会有影响),有资源申请时,一定要写,否则会造成资源泄漏

为什么析构不可以自己去处理内置类型呢?
因为内置类型中有指针等许多不能随便处理的类型,假如指针指向了一块空间,如果析构函数可以处理内置类型的话,有可能会直接把指针指向的空间给销毁了,这样指针就变成了野指针

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2024042002-计算机网络 - 应用层

计算机网络 - 应用层 计算机网络 - 应用层 域名系统文件传送协议动态主机配置协议远程登录协议电子邮件协议 1. SMTP2. POP33. IMAP 常用端口Web 页面请求过程 1. DHCP 配置主机信息2. ARP 解析 MAC 地址3. DNS 解析域名4. HTTP 请求页面 域名系统 DNS 是一个分布式数据库&…

24长三角数学建模ABC题已出!!!

需要ABC题资料的宝子们可以进企鹅 赛题如下&#xff1a; 赛道 A&#xff1a;“抢救”落水手机 上有天堂&#xff0c;下在苏杭&#xff1b;五一假期&#xff0c;杭州西湖、西溪湿地、京杭大运河等著名 景点&#xff0c;游人如织&#xff0c;作为享誉国内外的旅游胜地&#xff0…

华为OD机试 - 山峰个数(Java 2024 C卷 100分)

华为OD机试 2024C卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试&#xff08;JAVA&#xff09;真题&#xff08;A卷B卷C卷&#xff09;》。 刷的越多&#xff0c;抽中的概率越大&#xff0c;每一题都有详细的答题思路、详细的代码注释、样例测试…

Android Studio kotlin 转 Java

一. 随笔记录 java代码可以转化成kotlin代码&#xff0c;当然 Kotlin 反过来也可以转java 在Android Studio中 可以很方便的操作 AS 环境&#xff1a;Android Studio Iguana | 2023.2.1 二. 操作步骤 1.步骤 顶部Tools ----->Kotlin ------>Show Kotlin Bytecode 步…

【全开源】JAVA上门家政服务系统源码微信小程序+微信公众号+APP+H5

上门家政服务系统&#xff1a;便捷、专业&#xff0c;让家更温馨 随着现代生活节奏的加快&#xff0c;越来越多的人面临着忙碌的工作和紧张的生活压力&#xff0c;对于家庭事务的处理往往力不从心。为了解决这个问题&#xff0c;我们推出了全新的“上门家政服务系统”&#xf…

如何搜索空文件夹_名称为(纯或含)中/英/数/符

首先&#xff0c;需要用到的这个工具&#xff1a; 度娘网盘 提取码&#xff1a;qwu2 蓝奏云 提取码&#xff1a;2r1z 打开工具&#xff0c;切换到批量文件复制版块&#xff0c;快捷键Ctrl5 点击右侧的搜索添加 设定要搜索的范围、指定为文件夹、包括子目录&#xff0c;勾选详…

代码随想录——二叉树的最大深度(Leetcode104)

题目链接 层序遍历 当遍历到二叉树每一层最后一个节点时&#xff0c;depth /*** Definition for a binary tree node.* public class TreeNode {* int val;* TreeNode left;* TreeNode right;* TreeNode() {}* TreeNode(int val) { this.val val; }* …

【Linux】线程周边001之多线程

&#x1f440;樊梓慕&#xff1a;个人主页 &#x1f3a5;个人专栏&#xff1a;《C语言》《数据结构》《蓝桥杯试题》《LeetCode刷题笔记》《实训项目》《C》《Linux》《算法》 &#x1f31d;每一个不曾起舞的日子&#xff0c;都是对生命的辜负 目录 前言 1.线程的理解 2.地址…

如何在Sui智能合约中验证是否为多签地址

通过多签合约实现多个用户可访问的安全账户。多签&#xff08;multi-sig&#xff09;钱包和账户通过允许多个用户在预定义条件下访问共享资产&#xff0c;或让单个用户实施额外的安全措施&#xff0c;从而增强密钥管理。例如&#xff0c;多签钱包可以用于管理去中心化自治组织&…

bcb6 lib编程

Library 新建 Library 新建Cpp File File1.cpp extern "C" __declspec(dllexport) int add(int a,int b) {return ab;}Build Project->Build Project1 使用 新建项目 Add New Project Unit1.cpp #pragma hdrstop#include "Unit1.h" //---------…

企业微信hook接口协议,ipad协议http,客户群发送任务,获取要发送的客户群列表

客户群发送任务&#xff0c;获取要发送的客户群列表 参数名必选类型说明uuid是String每个实例的唯一标识&#xff0c;根据uuid操作具体企业微信 请求示例 {"uuid": "1688853790599424","id":110129274704417637, //群发任务id"keywords…

分布式锁:场景和使用方法(通俗讲解)

这里写目录标题 通俗讲解分布式锁&#xff1a;场景和使用方法前言引入业务场景业务场景一出现业务场景二出现&#xff1a;业务场景三出现&#xff1a; 分布式锁的使用场景分布式锁的几种特性分布式锁的几种实现方式一、基于 Mysql 实现分布式锁二、基于单Redis节点的分布式锁三…

HarmonyOS ArkTS 实现类似Android中RadioButton得效果

在Android中如实现下图可以用radioGroup和RadioButton实现&#xff0c;但在ArkTs中radio不能实现自定义样式&#xff0c;所以用Tabs来实现这种效果&#xff0c;效果图如下&#xff1a; 一、效果图 二、实现横向布局的三个TabContent&#xff0c;代码如下 State currentIndex: n…

Xilinx RAM IP核的使用及注意事项

对于RAM IP核(Block Memory Generator核)的使用以及界面的配置介绍&#xff0c;文章RAM的使用介绍进行了较详细的说明&#xff0c;本文对RAM IP核使用中一些注意的地方加以说明。 文章目录 三种RAM的区别单端口RAM(Single-port RAM)简单双端口RAM(Simple Dual-port RAM)真双端…

商品服务:SPUSKU规格参数销售属性

1.Object划分 1.PO&#xff08;Persistant Object&#xff09;持久对象 PO就是对应数据库中某个表中的一条记录&#xff0c;多个记录可以用PO的集合。PO中应该不报含任何对数据库的操作 2.DO(Domain Object) 领域对象 就是从现实世界中抽象出来的有形或无形的业务实体。 3…

线程池的简单实现与应用

1.什么是线程池 线程池其实就是一种多线程处理形式&#xff0c;处理过程中可以将任务添加到队列中&#xff0c;然后在创建线程后自动启动这些任务。 线程池最大的好处就是减少每次启动、销毁线程的损耗。 2.线程池参数介绍 参数名称说明corePoolSize正式员工的数量.(正式员…

代码随想录——填充每个节点的下一个右侧节点指针 II(Leetcode117)

题目链接 层序遍历 /* // Definition for a Node. class Node {public int val;public Node left;public Node right;public Node next;public Node() {}public Node(int _val) {val _val;}public Node(int _val, Node _left, Node _right, Node _next) {val _val;left _l…

21【Aseprite 作图】画白菜

1 对着参考图画轮廓 2 缩小尺寸 变成这样 3 本来是红色的描边&#xff0c;可以通过油漆桶工具&#xff08;取消 “连续”&#xff09;&#xff0c;就把红色的轮廓线&#xff0c;变成黑色的 同时用吸管工具&#xff0c;吸取绿色和白色&#xff0c;用油漆桶填充颜色 4 加上阴影…

svn批量解锁

问题 svn对文件进行checkout之后&#xff0c;先进行lock&#xff0c;之后再去更改&#xff0c;最后进行Commit操作&#xff1b; 上述为我们通过svn管理代码的正常方式&#xff0c;但总会有其他现象发生&#xff1b; 如果我们非正常操作&#xff0c;批量锁所有的svn文件&#x…

Django Celery 的配置及使用---最详细教程

Django Celery 的配置及使用 Redis提供队列消息功能 一、安装redis 系统版本&#xff1a;Ubuntu 20.041、获取最新软件包 sudo apt update sudo apt install redis-server2、安装完成后&#xff0c;Redis服务器会自动启动。查看redis是否启动成功 sudo systemctl status …