【C++】:构造函数和析构函数

目录

  • 前言
  • 一,构造函数
    • 1.1 什么是构造函数
    • 1.2 构造函数的特性
    • 1.3 总结
  • 二,析构函数
    • 2.1 什么是析构函数
    • 2.2 析构函数的特性
    • 2.3 总结

前言

如果一个类中什么成员都没有,简称为空类。

空类中真的什么都没有吗?并不是,任何类在什么都不写时,编译器会自动生成以下6个默认成员函数。

默认成员函数用户没有显式实现,编译器会生成的成员函数称为默认成员函数

在这里插入图片描述

这篇文章介绍的是构造函数析构函数

一,构造函数

在C语言中,我们平时在建立一个栈,或是写一些函数时,可能有时偶尔会忘记调用初始化函数,这样轻则会导致数据的随机值初始化,重则会导致程序的崩溃。

再比如对于Date类,可以通过 Init 公有方法给对象设置日期,但如果每次创建对象时都调用该方法设置信息,未免有点麻烦,那能否在对象创建时,就将信息设置进去呢?

在C++中,就改进了一种方法来解决这个问题,就是构造函数。

1.1 什么是构造函数

构造函数是一个特殊的成员函数名字与类名相同,创建类类型对象时由编译器自动调用,以保证每个数据成员都有 一个合适的初始值,并且在对象整个生命周期内只调用一次

1.2 构造函数的特性

构造函数是特殊的成员函数,需要注意的是,构造函数虽然名称叫构造,但是构造函数的主要任务并不是开空间创建对象,而是初始化对象。(功能类似于 Init 函数)

1.2.1 函数名与类名相同。

1.2.2 无返回值,并且不要写 void。

1.2.3 对象实例化时编译器自动调用对应的构造函数。

1.2.4 构造函数可以重载。(意思是构造函数可以有多个,多种初始化方式。)

(1) 自己写的无参(没有传参数)构造函数:

注意:
1. 这里的无参构造 对象前面没有括号,为了跟函数声明区分。
2. 当没有传参数时,此时C++编译器会不做处理,默认把内置类型初始化为随机值。

 class Date{public:// 1.无参构造函数Date(){}void Print()
{cout << _year << "-" << _month << "-" << _day << endl;
}private:int _year;int _month;int _day;};int main()
{//注意:这里的无参构造 对象前面没有括号,为了跟函数声明区分//Date d1(); //errDate d1;  //调用无参构造函数d1.Print();return 0;
}

在这里插入图片描述

(2) 自己写的带参构造函数:

  • 注意:调用带参构造函数时,是在对象后跟参数列表。
class Date{public:// 2.带参构造函数Date(int year, int month, int day)
{_year = year;_month = month;_day = day;
}void Print()
{cout << _year << "-" << _month << "-" << _day << endl;
}private:int _year;int _month;int _day;};int main()
{Date d2( 2024, 4, 22);//调用带参构造函数d2.Print();return 0;
}

在这里插入图片描述

(3) 自己写的全缺省构造函数:

注意:
1. 全缺省构造函数与无参构造函数也是重载关系,但是两者只能调用其一,否则调用时会产生歧义
2. 写构造函数时,一般喜欢写全缺省,因为很好用。

class Date{public://3.全缺省构造函数//注意:与无参的构造函数也是重载,但调用时会衬产生歧义
Date(int year =1 , int month =1 , int day = 1 )
{_year = year;_month = month;_day = day;
}void Print()
{cout << _year << "-" << _month << "-" << _day << endl;
}private:int _year;int _month;int _day;};int main()
{Date d1;d1.Print();return 0;
}

在这里插入图片描述

1.2.5 如果类中没有显式定义(自己写的)构造函数,则C++编译器会自动生成一个无参的默认构造函数,一旦用户显式定义编译器将不再生成。

注意:
1. C++编译器会自动生成一个无参的默认构造函数,默认把内置类型初始化为随机值。
2. 如果此时存在自己写的带参数的构造函数,代码编译失败,因为一旦显式定义任何构造函数,编译器将不再生成。

class Date{public:void Print()
{cout << _year << "-" << _month << "-" << _day << endl;
}private:int _year;int _month;int _day;};int main()
{Date d1;d1.Print();return 0;
}

在这里插入图片描述

1.2.6 关于编译器生成的默认成员函数,很多童鞋会有疑惑:不实现构造函数的情况下,编译器会生成默认的构造函数。但是看起来默认构造函数又没什么用?d对象调用了编译器生成的默认构造函数,但是d对象_year/_month/_day,依旧是随机值。也就说在这里编译器生成的默认构造函数并没有什么用??

解答:

  • C++把类型分成内置类型(基本类型)和自定义类型。内置类型就是语言提供的数据类型,如:int/char…,自定义类型就是我们使用class/struct/union等自己定义的类型。对于内置类型,在刚开始发明C++时,并没有规定要不要处理(有些编译器会初始化为0,有些则是初始化为随机值),对于自定义类型的成员变量才会调用它们的无参构造

看看下面的程序,Date类里面包含了内置类型和自定义类型,Date类中会自动调用无参构造函数把它的内置类型成员变量初始化为随机值,并且对于自定义类型成员变量 _aa会自动调用它的无参构造。

class A
{
public://err  当没有无参构造时,就会报错/*A(int a){_a = 0;cout << "A()" << endl;}*///如果连显示写的无参构造函数都没有,也会有编译器默认的无参构造//此时也要看这里有没有自定义类型(就像套娃),这里只有内置类型,所以还是不做处理,初始化随机值A(){_a = 0;cout << "A()" << endl;}private:int _a;};class Date
{
public:Date(){}void Print(){cout << _year << "-" << _month << "-" << _day << endl;}private:// 基本类型(内置类型)int _year;  // 年int _month; // 月int _day;   // 日// 自定义类型A _aa;
};int main()
{Date d;d.Print();return 0;
}

在这里插入图片描述

注意:针对编译器给内置类型初始化随机值的问题,C++11 中针对内置类型成员不初始化的缺陷,又打了补丁,即:内置类型成员变量在类中声明时可以给缺省值

class Date
{
public:void Print(){cout << _year << "-" << _month << "-" << _day << endl;}private://给缺省值int _year =1;  // 年int _month =1; // 月int _day =1;   // 日
};int main()
{Date d;d.Print();return 0;
}

在这里插入图片描述

1.2.7 无参的构造函数全缺省的构造函数都称为默认构造函数并且默认构造函数只能有一个。

注意:无参构造函数、全缺省构造函数、我们没写编译器默认生成的构造函数,都可以认为是默认构造函数。三者只能存其一。

比如,当同时存在其中的两个时:程序报错!

class Date
{
public:Date(){}Date(int year = 1, int month = 1, int day = 1){_year = year;_month = month;_day = day;}void Print(){cout << _year << "-" << _month << "-" << _day << endl;}private:int _year ;  // 年int _month ; // 月int _day ;   // 日
};int main()
{Date d;d.Print();return 0;
}

在这里插入图片描述

1.3 总结

1.一般情况下构造函数都需要我们自己显式的去实现
2.只有少数情况下可以让编译器自动生成构造函数。(类似用两个栈实现队列的MyQueue,它的成员都是自定义类型)

二,析构函数

通过前面构造函数的学习,我们知道一个对象是怎么来的,那一个对象又是怎么没呢的?

2.1 什么是析构函数

析构函数:与构造函数功能相反,析构函数不是完成对对象本身的销毁,局部对象销毁工作是其出了所在域之后栈帧销毁时完成的。而对象在销毁时会自动调用析构函数,完成对象中资源的清理工作(一般是清理动态开辟,文件打开,new等的资源)

注:析构函数也可以显式调用(手动调用),相当于Destroy 两次。

2.2 析构函数的特性

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

2.2.1 析构函数名是在类名前加上字符 ~。

2.2.2 无参数无返回值类型

2.2.3 一个类只能有一个析构函数。若未显式定义,系统会自动生成默认的析构函数。注意:析构函数不能重载。(因为重载的本质是同名函数的参数差异,而析构函数无参)

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

看看下面的自己写的显式析构函数:

typedef int DataType;
class Stack
{
public://构造函数Stack(size_t capacity = 3){cout << "Stack(size_t capacity = 3)" << endl;_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(){cout << "~Stack()" << endl;if (_array){free(_array);_array = NULL;_capacity = 0;_size = 0;}}private:DataType* _array;int _capacity;int _size;
};int main()
{Stack st;return 0;
}

![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/75aa79f162db41f2bbbdd4440b97653f.png

注意:当进行多文件操作,函数的定义和声明分离时,要在定义函数时指定类域。

在这里插入图片描述

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

class Time
{
public:~Time(){cout << "~Time()" << endl;}
private:int _hour;int _minute;int _second;
};class Date
{
private:// 基本类型(内置类型)int _year = 1970;int _month = 1;int _day = 1;// 自定义类型Time _t;
};int main()
{Date d;return 0;
}

在这里插入图片描述

在main方法中根本没有直接创建Time类的对象,为什么最后会调用Time类的析构函数?

因为:main方法中创建了Date对象d,而d中包含4个成员变量,其中_year, _month, _day三个是内置类型成员,销毁时不需要资源清理,最后系统直接将其内存回收即可;

而_t是Time类对象,所以在d销毁时,要将其内部包含的Time类的_t对象销毁,所以要调用Time类的析构函数。

但是:main函数中不能直接调用Time类的析构函数,实际要释放的是Date类对象,所以编译器会调用Date类的析构函数,而Date没有显式提供,则编译器会给Date类生成一个默认的析构函数,目的是在其内部调用Time类的析构函数,即当Date对象销毁时,要保证其内部每个自定义对象都可以正确销毁main函数中并没有直接调用Time类析构函数,而是显式调用编译器为Date类生成的默认析构函数。

注意:创建哪个类的对象则调用该类的析构函数,销毁那个类的对象则调用该类的析构函数。

2.3 总结

1.有资源需要清理的,就需要自己写析构函数。如Stack,List……

2.以下两种场景不需要写析构函数,用自动生成的就可以:
(1) 没有资源需要清理。如Date类;
(2) 内置成员类型没有资源需要清理,剩下的都是自定义类型成员。如MyQueue。

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

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

相关文章

深度学习中的子空间、线性变换和矩阵概念应用

1.表示子空间 在深度学习中&#xff0c;“不同的表示子空间”通常是指模型通过不同的参数&#xff08;例如权重矩阵&#xff09;将输入数据映射到不同的高维空间&#xff0c;这些空间被称为表示子空间。每个子空间都能够捕获输入数据中不同的特征或模式。以下是一些详细解释&am…

新手Pytorch入门笔记-transforms.Compose()

我使用的图片是上图&#xff0c;直接下载即可 transforms.Compose 是PyTorch中的一个实用工具&#xff0c;用于创建一个包含多个数据变换操作的变换对象。这些变换操作通常用于数据预处理&#xff0c;例如图像数据的缩放、裁剪、旋转等。使用transforms.Compose 可以将多个数据…

广工电工与电子技术实验报告-8路彩灯循环控制电路

实验代码 module LED_water (clk,led); input clk; output [7:0] led; reg [7:0] led; integer p; reg clk_1Hz; reg [7:0] current_state, next_state; always (posedge clk) begin if(p25000000-1)begin …

接口测试和Mock学习路线(上)

一、接口测试和Mock学习路线-第一阶段&#xff1a; 掌握接口测试的知识体系与学习路线掌握面试常见知识点之 HTTP 协议掌握常用接口测试工具 Postman掌握常用抓包工具 Charles 与 Fiddler结合知名产品实现 mock 测试与接口测试实战练习 1.接口协议&#xff1a; 需要先了解 O…

微服务组件-反向代理(Nginx)

微服务组件-反向代理(Nginx) Nginx 基本概念 1、nginx是什么&#xff1f; ①、Nginx (engine x) 是一个高性能的HTTP和反向代理web服务器同时也提供了IMAP/POP3/SMTP服务。它是一款轻量级的Web服务器/反向代理服务器及电子邮件&#xff08;IMAP/POP3&#xff09;代理服务器&a…

Java | Leetcode Java题解之第50题Pow(x,n)

题目&#xff1a; 题解&#xff1a; class Solution {public double myPow(double x, int n) {long N n;return N > 0 ? quickMul(x, N) : 1.0 / quickMul(x, -N);}public double quickMul(double x, long N) {if (N 0) {return 1.0;}double y quickMul(x, N / 2);retu…

【UE5】蓝图通信方式

目录 1、直接通信 2、getAllActorsOfClass 3、getAllActorsOfClassWithTag 4、通过射线检测 5、接口 6、事件分发器 7、SpawnActor 8、调用控制台命令 9、关卡蓝图中直接调用 创建两个Actor蓝图 1、直接通信 场景中 2、getAllActorsOfClass 3、getAllActorsOfClassWit…

学习Rust第14天:HashMaps

今天我们来看看Rust中的hashmaps&#xff0c;在 std::collections crate中可用&#xff0c;是存储键值对的有效数据结构。本文介绍了创建、插入、访问、更新和迭代散列表等基本操作。通过一个计算单词出现次数的实际例子&#xff0c;我们展示了它们在现实世界中的实用性。Hashm…

故障诊断 | 基于迁移学习和SqueezeNet 的滚动轴承故障诊断(Matlab)

目录 效果一览基本介绍程序设计参考文献 效果一览 基本介绍 将一维轴承振动信号转换为二维尺度图&#xff08;时频谱图&#xff09;&#xff0c;并使用预训练网络应用迁移学习对轴承故障进行分类。 迁移学习显著减少了传统轴承诊断方法特征提取和特征选择所花费的时间&#xff…

Coursera: An Introduction to American Law 学习笔记 Week 02: Contract Law

An Introduction to American Law 本文是 https://www.coursera.org/programs/career-training-for-nevadans-k7yhc/learn/american-law 这门课的学习笔记。 文章目录 An Introduction to American LawInstructors Week 02: Contract LawKey Contract Law TermsSupplemental Re…

C语言笔试题之计数质数

计数质数 实例要求 给定整数 n &#xff0c;返回 所有小于非负整数 n 的质数的数量&#xff1b;示例&#xff1a; 实例分析 1、要计算小于非负整数 n 的质数的数量&#xff0c;可以使用埃拉托斯特尼筛法&#xff1b;2、这个算法通过标记素数的倍数来找出所有的素数&#x…

RTK负载(4K可见光+高分热成像+超广角+激光测距)四光AI智能识别跟踪吊舱技术详解

无人机光电吊舱的RTK负载&#xff08;4K可见光高分热成像超广角激光测距&#xff09;AI智能识别跟踪吊舱技术是一种高度集成和先进的无人机观测系统。系统结合了无人机的飞行能力和光电吊舱的多功能传感器&#xff0c;通过集成RTK&#xff08;实时动态差分定位&#xff09;技术…

STL_deque_stack_queue

Deque deque容器(双端队列) ​deque是一种双向开口的分段连续线性空间&#xff08;对外号称连续&#xff0c;使用者无法感知它是分段的&#xff09;。deque支持从头尾两端进行元素的插入和删除。deque没有容量的概念&#xff0c;因为它是动态地以分段连续空间组合而成的。随时…

python 脚本头(PyCharm+python头部信息、py头部信息、python头信息、py头信息、py文件头部)

文章目录 参考PyCharm设置脚本头头部信息 参考 https://developer.aliyun.com/article/1166544 https://blog.csdn.net/Dontla/article/details/131743495 https://blog.csdn.net/dongyouyuan/article/details/54408413 PyCharm设置脚本头 打开pycharm&#xff0c;点击file–…

【Harmony3.1/4.0】笔记六-对话框

概念 对话框在任何一款应用中&#xff0c;任何一个系统或者平台上使用都非常频繁&#xff0c;这里介绍一下鸿蒙系统中对话框的用法&#xff0c;分别为:普通文本对话框&#xff0c;自定义提示对话框&#xff0c;对话框菜单&#xff0c;警告提示对话框&#xff0c;列表选择对话框…

Unity 实现原神中的元素反应

一、元素反应 原神中共有七种元素&#xff0c;分别是水、火、冰、岩、风、雷、草。这七种元素能互相作用 Demo下载&#xff1a;Download 元素反应表格图示&#xff0c;可能不够精准 /火水雷冰草岩风绽放原激化火/蒸发超载融化燃烧结晶扩散烈绽放/水蒸发/感电冻结/碎冰绽放结晶…

Windows主机入侵检测与防御内核技术深入解析

第2章 模块防御的设计思想 2.1 执行与模块执行 本章内容为介绍模块执行防御。在此我将先介绍“执行”分类&#xff0c;以及“模块执行”在“执行”中的位置和重要性。 2.1.1 初次执行 恶意代码&#xff08;或者行为&#xff09;要在被攻击的机器上执行起来&#xff0c;看起…

Ubuntu 自己写的程序如何创建快捷方式

在Ubuntu中创建程序的快捷方式通常是通过将一个指向程序可执行文件的.desktop文件放入/usr/share/applications/或用户的~/.local/share/applications/目录来实现的。以下是创建快捷方式的基本步骤和示例&#xff1a; 在application里创建快捷方式 创建一个新的.desktop文件。…

【Linux】详解信号产生的方式

一、kill命令 在命令行中通过kill -数字 pid指令可以给指定进程发送指定信号。这里说明一下几个常见的信号&#xff1a; SIGINT&#xff08;2号信号&#xff09;&#xff1a;中断信号&#xff0c;通常由用户按下CtrlC产生&#xff0c;用于通知进程终止。SIGQUIT&#xff08;3号…

PG修改端口号与error: could not connect to server: could not connect to server 问题解决

刚开始学习PG修改端口号之后数据库端口号没变。 修改端口号&#xff1a;/usr/local/pgsql/data中的postgresql.conf中 修改后并不能直接生效需要重启PG&#xff1a; /usr/local/pgsql/bin/pg_ctl -D /usr/local/pgsql/data -l /usr/local/pgsql/data/logfile restart重启后新…