【浅尝C++】C++类的6大默认成员函数——构造、析构及拷贝构造函数

在这里插入图片描述

🎈归属专栏:浅尝C++
🚗个人主页:Jammingpro
🐟记录一句:好想摆烂,又好想学习~~


文章前言:本篇文章简要介绍C++类的构造函数、析构函数及拷贝构造函数,介绍每个小点时,都会附上对应的代码,如果可能的话。余下3大默认成员函数将于后续文章中叙述,要不,整个文章太长了。


文章目录

  • 构造函数
    • 基本概念
    • 特性
  • 析构函数
    • 基本概念
    • 特性
  • 拷贝构造函数
    • 概念
    • 特征


我们知道,如果一个类什么都没有,那么它就是一个空类。空类真的什么都没有吗?答案是否定的。任何类在什么都不写时,编译器会自动生成以下6个默认成员函数,本文将先介绍构造函数、析构函数及拷贝构造函数。👇
ps:默认成员函数:用户没有显式实现,编译器会生成的成员函数称为默认成员函数。
在这里插入图片描述

构造函数

基本概念

在介绍构造函数之前,我们先回顾一下类的定义和调用👇。

#include<iostream>
using namespace std;class Date
{
public:void Init(int year, int month, int day){m_year = year;m_month = month;m_day = day;}void Print(){cout << m_year << " - " << m_month << " - " << m_day << endl;}
private:int m_year;int m_month;int m_day;
};int main()
{Date date;date.Init(2023, 1, 1);date.Print();return 0;
}

对于Date类,可以通过 Init 公有方法给对象设置日期,但如果每次创建对象时都调用该方法设置信息,未免有点麻烦,那能否在对象创建时,就将信息设置进去呢?
构造函数是一个特殊的成员函数,名字与类名相同,创建类类型对象时由编译器自动调用,以保证每个数据成员都有 一个合适的初始值,并且在对象整个生命周期内只调用一次

特性

构造函数是特殊的成员函数,需要注意的是,构造函数虽然名称叫构造,但是构造函数的主要任务并不是开空间创建对象,而是初始化对象
特征如下:
①函数名与类名相同。
②无返回值。
③对象实例化时编译器自动调用对应的构造函数。
④构造函数可以重载。

#include<iostream>
using namespace std;class Date
{
public://以下两个构造函数,参数个数不同,构成重载Date(int year, int month, int day){//定义对象时,会自动打印下面这句内容cout << "Date(int year, int month, int day)被调用" << endl;m_year = year;m_month = month;m_day = day;}//构造函数名称与类名相同,且无返回值Date(int year){cout << "Date(int year)被调用" << endl;}
private:int m_year;int m_month;int m_day;
};int main()
{Date d1(2022, 12, 12);Date d3(2023);return 0;
}

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

#include<iostream>
using namespace std;class Date
{
public:/*提供该构造函数后,编译不再生成默认构造函数Date(int year, int month, int day){m_year = year;m_month = month;m_day = day;}*/
private:int m_year;int m_month;int m_day;
};int main()
{Date d1;//若将上面类中定义的构造函数注释掉,则此处不会出错,因为编译器自动生成了默认构造函数(不带参数的函数)//若使用上面类中定义的构造函数,则此处定义d1对象会报错,因为编译器不再自动生成默认构造函数return 0;
}

⑥关于编译器生成的默认成员函数,不实现构造函数的情况下,编译器会生成默认的构造函数。但是看起来默认构造函数又没什么用?d对象调用了编译器生成的默认构造函数,但是d1对象m_year/m_month/m_day,依旧是随机值。也就说在这里编译器生成的默认构造函数并没有什么用??
答:C++把类型分成内置类型(基本类型)和自定义类型。内置类型就是语言提供的数据类型,如:int/char…,自定义类型就是我们使用class/struct/union等自己定义的类型,看看下面的程序,就会发现编译器生成默认的构造函数会对自定类型成员m_t调用的它的默认成员函数

#include<iostream>
using namespace std;class Time
{
public:Time(){cout << "Time构造函数" << endl;m_hour = m_minute = m_second = 0;}void Print(){cout << m_hour << ":" << m_minute << ":" << m_second << endl;}
private;int m_hour;int m_minute;int m_second;
};class Date
{
public:void Print(){cout << m_year << "-" << m_month << "-" << m_day << endl;m_t.Print();}
private:int m_year;int m_month;int m_day;
};int main()
{Date d;//d未调用Time的构造函数,但打印时,Time的各个成员变量均已初始化为1。说明编译器自动实现的Date的构造函数,会自动调用Time的构造函数。d.Print();return 0;
}

注意:C++11 中针对内置类型成员不初始化的缺陷,又打了补丁,即:内置类型成员变量在类中声明时可以给默认值

#include<iostream>
#include<string>
using namespace std;class Jammingpro
{
public:void Print(){cout << "我是 " << m_name << ",今年 " << m_age << " 岁" << endl;}
private:string m_name = "xioaming";int m_age = "18";
};int main()
{Jammingpro jam;jam.Print();//打印的是默认值return 0;
}

⑦无参的构造函数和全缺省的构造函数都称为默认构造函数,并且默认构造函数只能有一个。
注意:无参构造函数、全缺省构造函数、我们没写编译器默认生成的构造函数,都可以认为是默认构造函数。下面代码演示的是无参、全缺省构造函数,两个函数只能留下其中一个,不能同时存在。

class Date
{
public:Date(){}Date(int year = 2022, int month = 12, int day = 12){m_year = year;m_month = month;m_day = day;}
private:int m _year;int m_month;int m_day;
}

析构函数

基本概念

与构造函数功能相反,析构函数是完成对对象本身的销毁,局部对象销毁工作是由编译器完成的。而对象在销毁时会自动调用析构函数,完成对象中资源的清理工作。

特性

析构函数是特殊的成员函数,其特征如下:
①析构函数名是在类名前加上字符 ~。
②无参数无返回值类型。
③一个类只能有一个析构函数。若未显式定义,系统会自动生成默认的析构函数。注意:析构函数不能重载
④对象生命周期结束时,C++编译系统系统自动调用析构函数。

#include<iostream>
using namespace std;class Stack
{
public://对象生命周期结束时,需要调用下面的析构函数~Stack(){cout << "~Stack()" << endl;}
private:char* m_mem;int m_size;int m_capacity;
};int main()
{Stack stk;return 0;
}

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

#include<iostream>
using namespace std;class Time
{
public://下面这句话会被打印,因为Date的自动生成的析构函数会调用自定义数据类型的析构函数~Time(){cout << "Time析构函数" << endl;}
private;int m_hour;int m_minute;int m_second;
};class Date
{
private:int m_year;int m_month;int m_day;
};int main()
{Date d;return 0;
}

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

class Stack 
{
public:Stack(){m_mem = new char[100];//在堆区申请了内存空间m_size = 0;m_capcpity;}~Stack(){delete[] m_mem;//必须使用析构函数释放资源}
private:char* m_mem;int m_size;int m_capacity;
}

拷贝构造函数

概念

拷贝构造函数:只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),在用已存在的类类型对象创建新对象时由编译器自动调用。

特征

其特征如下:
①拷贝构造函数是构造函数的一个重载形式。
②拷贝构造函数的参数只有一个且必须是类类型对象的引用,使用传值方式编译器直接报错,因为会引发无穷递归调用

class Date
{
public:Date(int year, int month, int day){m_year = year;m_month = month;m_day = day;}//Date(const Date d) ->错误写法,将引发无穷递归Date(const Date& d)//->正确写法{m_year = d.m_year;m_month = d.m_month;m_day = d.m_day;}
private:int m_year;int m_month;int m_day;
}

③若未显式定义,编译器会生成默认的拷贝构造函数。 默认的拷贝构造函数对象按内存存储按字节序完成拷贝,这种拷贝叫做浅拷贝,或者值拷贝。
注意:在编译器生成的默认拷贝构造函数中,内置类型是按照字节方式直接拷贝的,而自定义类型是调用其拷贝构造函数完成拷贝的。

class Stack
{
public:Stack(){m_mem = new char[100];m_size = 0;m_capacity = 100;}/*下面这个是编译器自动生成的Stack(const Stack& s){m_mem = s.m_mem;m_size = s.m_size;m_capacity = s.m_capacity;}*/Stack(const Stack& s){m_mem = new char[s.m_capacity];strcpy(m_mem, s.m_mem);m_size = s.m_size;m_capacity = s.m_capacity;}
private:char* m_mem;int m_size;int m_capacity;
}

④编译器生成的默认拷贝构造函数已经可以完成字节序的值拷贝了,还需要自己显式实现吗?当然像Date类这样的类是没必要的。而像上面的Stack类是需要自己显示实现拷贝构造函数的。
编译器自动生成的拷贝构造函数是浅拷贝,也就是将属性值复制一份。对于下面图片中的s1s2,由于s2复制了s1m_mem的地址,导致两个栈指向同一内存空间,两个栈生命周期结束后均会调用析构函数,此时将执行两次delete[] m_mem,将导致内存重复释放而报错。
注意:类中如果没有涉及资源申请时,拷贝构造函数是否写都可以;一旦涉及到资源申请时,则拷贝构造函数是一定要写的,否则就是浅拷贝。
在这里插入图片描述
⑤拷贝构造函数典型调用场景:
使用已存在对象创建新对象
函数参数类型为类类型对象
函数返回值类型为类类型对象

#include<iostream>
using namespace std;class Date
{
public:Date(int year, int month, int day){cout << "Date create" << endl;m_year = year;m_month = month;m_day = day;}Date(const Date& d){cout << "Date copy" << endl;m_year = d.m_year;m_month = d.m_month;m_day = d.m_day;}~Date(){cout << "~Date()" << endl;}
private:int m_year;int m_month;int m_day;
};Date Test(Date d)
{return d;
}int main()
{Date date(2023, 11, 11);Test(date);Date cp(date);return 0;
}

这段代码的执行结果如下,即调用1次构造函数,3次拷贝构造函数,4次析构函数:

Date create
Date copy
Date copy
~Date()
~Date()
Date copy
~Date()
~Date()

对于上面的代码,执行Date date(2023, 11, 11);时调用1次构造函数;执行Test(date);时传递参数,调用了1次拷贝构造函数,返回时,返回值又调用了1次拷贝构造函数,由于返回值产生的临时变量和Test的参数d离开Test函数后生命周期就结束了,因此分别调用了析构函数,即2次析构函数;执行Date cp(date)构造cp对象时调用了1次拷贝构造函数,main函数执行结束,date与cp生命周期结束,分别调用析构函数,即2次析构函数。
为了提高程序效率,一般对象传参时,尽量使用引用类型,返回时根据实际场景,能用引用尽量使用引用


文章结语:这篇文章对C++中的构造函数、析构函数及拷贝构造函数进行了简要的介绍。后续文章将介绍余下3大默认成员函数。
🎈欢迎进入浅尝C++专栏,查看更多文章。
如果上述内容有任何问题,欢迎在下方留言区指正b( ̄▽ ̄)d

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

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

相关文章

C语言公交车之谜(ZZULIOJ1232:公交车之谜)

题目描述 听说郑州紫荆山公园有英语口语角&#xff0c;还有很多外国人呢。为了和老外对上几句&#xff0c;这周六早晨birdfly拉上同伴早早的就坐上了72路公交从学校向紫荆山进发。一路上没事干&#xff0c;birdfly开始思考一个问题。 从学校到紫荆山公园共有n(1<n<20)站路…

Git本地库操作

对本地库的操作很少&#xff0c;我们学习1~6节即可&#xff0c;其他了解下。我们可以在idea中完成对本地库还有远程库的操作&#xff0c;可视化界面用起来更加舒适而且也不会混淆。 1. Git概述 Git 是一个免费的、开源的分布式版本控制系统&#xff0c;可以快速高效地处理从小…

Cesium 可视化深度纹理

Cesium 可视化深度纹理 // 创建纹理辅助器图元const textureHelper new TextureHelperPrimitive(viewer.scene)viewer.scene.primitives.add(textureHelper)viewer.scene.postRender.addEventListener(function () {const framebuffer viewer.scene.view.pickDepths[0]?.fra…

问题:vue2+elementui,tabs切换显示表格并设置表格选中行高亮失败

错误示范&#xff1a; 1.直接setCurrentRow失败&#xff08;this.currentRow是之前保存的表格当前选中行的数据&#xff09; this.$refs.table.setCurrentRow(this.currentRow);2.以为是表格没生成就执行了setCurrentRow导致设置不成功&#xff0c;所以使用了this.$nextTick&…

[网鼎杯 2020 朱雀组]Nmap

启动环境 结合题目首先就是要知道关于关于nmap命令 相关的命令-oN 标准保存 -oX XML保存 -oG Grep保存 -oA 保存到所有格式 -iL 读取文件内容&#xff0c;以文件内容作为搜索目标 -o 输出到文件 -sP Ping 扫描 还有许多 nmap命令https://blog.csdn.net/weixin_735627…

30.0/集合/ArrayList/LinkedList

目录 30.1什么是集合? 30.1.2为什么使用集合 30.1.3自己创建一个集合类 30.1.3 集合框架有哪些? 30.1.2使用ArrayList集合 30.2增加元素 30.3查询的方法 30.4删除 30.5 修改 30.6泛型 30.1什么是集合? 我们之前讲过数组&#xff0c;数组中它也可以存放多个元素。集合…

vue3 for循环创建的多个e-form 添加校验

v-for 创建 ref <el-form :model"item" :rules"state.rules" :ref"el > getRiskSpreadRef(el, index)" ></el-form>// 定义ref list const riskSpreadRefList ref<HTMLElement[]>([]);// ref存到数组 const getRiskSpread…

前端 计算机基础篇 ( 二 )

文章目录 websockt及原理ipv4和ipv6的区别线程和进程的区别cdn原理缓存所涉及的http状态码缓存的时候设置 no-store和no-cache和max-age0这几个有什么区别token一般存放在哪儿怎么设置强缓存和协商缓存强缓存&#xff1a;1. 使用 Cache-Control 头字段&#xff1a; 协商缓存&am…

竞赛选题 题目:基于LSTM的预测算法 - 股票预测 天气预测 房价预测

文章目录 0 简介1 基于 Keras 用 LSTM 网络做时间序列预测2 长短记忆网络3 LSTM 网络结构和原理3.1 LSTM核心思想3.2 遗忘门3.3 输入门3.4 输出门 4 基于LSTM的天气预测4.1 数据集4.2 预测示例 5 基于LSTM的股票价格预测5.1 数据集5.2 实现代码 6 lstm 预测航空旅客数目数据集预…

使用SpringBoot集成MyBatis对管理员的查询操作

增删改查中的查询操作&#xff0c;对所有的普通管理员进行查询操作。 效果展示&#xff1a; 不仅可以在打开页面时进行对管理员的自动查询操作&#xff0c;还可以在输入框进行查询。 首先是前端向后端发送POST请求&#xff0c;后端接收到请求&#xff0c;如果是有参数传到后端…

C语言--每日选择题--Day27

第一题 1. 对于代码段&#xff0c;问下面不可以表示a[1]地址的是&#xff08;&#xff09; int a[10]; A&#xff1a;&a[0] 1 B&#xff1a;a sizeof(int) C&#xff1a;(int*)&a 1 D&#xff1a;(int*)((char*)&a sizeof(int)) 答案及解析 A A&#xff1a;取到…

免费获取GPT-4的五种工具

不可否认&#xff0c;由OpenAI带来的GPT-4已是全球最受欢迎的、功能最强大的大语言模型&#xff08;LLM&#xff09;之一。大多数人都需要使用ChatGPT Plus的订阅服务去访问GPT-4。为此&#xff0c;他们通常需要每月支付20美元。那么问题来了&#xff0c;如果您不想每月有这笔支…

redis运维(十五) 集合

一 集合 ① 概念 集合的元素在redis里面的世界是member集合&#xff1a; setset集合当中不允许重复的元素&#xff0c;而且set集合当中元素是没有顺序的,不存在元素下标 ② sadd、smembers、srem ③ sismember、srandmember、spop、scard spop 命令用于移除集合中的指定 …

docker国内镜像加速

创建或修改 /etc/docker/daemon.json 文件&#xff0c;修改为如下形式 {"registry-mirrors": ["https://registry.docker-cn.com","http://hub-mirror.c.163.com","https://docker.mirrors.ustc.edu.cn"] } Docker中国区官方镜像htt…

MidJourney笔记(4)-settings

前面已经大概介绍了MidJourney的基础知识,后面我主要是基于实操来分享自己的笔记。可能内容顺序会有点乱,请大家理解。 这次主要是想讲讲settings这个命令。我们只需在控制台输入/settings,然后回车,就可以执行这个命令。 (2023年11月26日版本界面) 可能有些朋友出来的界…

C# Winform使用log4net记录日志

写在前面 Log4Net是从Java的log4j移植过来的&#xff0c;功能也与log4j类似&#xff0c;可以把日志信息输出到文件、数据库、控制台、Windows 事件日志、远程系统日志服务等不同的介质或目标。 Log4Net配置选项丰富灵活&#xff0c;并且可在运行时动态更新配置并应用&#xf…

springboot+vue智能企业设备管理系统05k50

智能设备管理系统主要是为了提高工作人员的工作效率和更方便快捷的满足用户&#xff0c;更好存储所有数据信息及快速方便的检索功能&#xff0c;对系统的各个模块是通过许多今天的发达系统做出合理的分析来确定考虑用户的可操作性&#xff0c;遵循开发的系统优化的原则&#xf…

【python+requests】接口自动化测试

这两天一直在找直接用python做接口自动化的方法&#xff0c;在网上也搜了一些博客参考&#xff0c;今天自己动手试了一下。 一、整体结构 上图是项目的目录结构&#xff0c;下面主要介绍下每个目录的作用。 Common:公共方法:主要放置公共的操作的类&#xff0c;比如数据库sql…

【开源】基于Vue和SpringBoot的独居老人物资配送系统

项目编号&#xff1a; S 045 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S045&#xff0c;文末获取源码。} 项目编号&#xff1a;S045&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块三、系统展示四、核心代码4.1 查询社区4…

学生信息管理系统程序Python

系统主界面 在该界面中可以选择要使用功能对应的菜单进行不同的操作。在选择功能菜单时&#xff0c;有两种方法&#xff0c; 一种是输入1&#xff0c;另一种是按下键盘上的↑或↓方向键进行选择。这两种方法的结果是一样的&#xff0c;所以使用哪种方法都可以。 &#xff08;…