【浅尝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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

java+python农村集体产权管理系统php+vue

注册、登陆该系统根据操作权限的不同分为管理员和用户两种&#xff0c;新用户在登陆前要进行用户注册&#xff0c;注册完成后方可进行登陆。 本次设计的关键问题处理&#xff0c;主要有如下几点&#xff1a; (1&#xff09;本次开发&#xff0c;采用主流Thinkphp框架进行开发&a…

linux进入telnet和推出telnet

安装telnet centos7 yum install -y telnet ubuntu apt install -y telnet 进入telnet telnet ip port 退出telnet 1. 按下下面的组合键 ctrl] 2. 输入下面命令推出 quit

Go 语言中 sync 包的近距离观察

让我们来看看负责提供同步原语的 Go 包&#xff1a;sync。 sync.Mutex sync.Mutex 可能是 sync 包中被广泛使用的原语。它允许对共享资源进行互斥操作&#xff08;即不允许同时访问&#xff09;&#xff1a; mutex : &sync.Mutex{}mutex.Lock() // Update shared variab…

【linux】基本指令(中篇)

echo指令 将引号内容打印到显示屏上 输出的重定向 追加的重定向 输出的重定向 我们学习c语言的时候当以写的方式创建一个文件&#xff0c;就会覆盖掉该文件之前的内容 当我们以追加的方式打开文件的时候&#xff0c;原文件内容不会被覆盖而是追加 more指令 10.more指令…

YOLOv8优化策略:自适应改变核大小卷积AKConv,效果优于标准卷积核和DSConv |2023.11月最新成果

🚀🚀🚀本文改进: AKConv 中,通过新的坐标生成算法定义任意大小的卷积核的初始位置。 为了适应目标的变化,引入了偏移量来调整每个位置的样本形状。 此外,我们通过使用具有相同大小和不同初始采样形状的 AKConv 来探索神经网络的效果。 AKConv 通过不规则卷积运算完成…

简介vue

目录 一、介绍 渐进式框架​ 单文件组件​ 选项式 API (Options API)​ 组合式 API (Composition API)​ 该选哪一个&#xff1f;​ 创建一个 Vue 应用 应用实例​ 根组件​ DOM 中的根组件模板 应用配置​ 多个应用实例​ 一、介绍 Vue (发音为 /vjuː/&#xff…

代码随想录算法训练营第四十六天|139.单词拆分、背包问题总结

LeetCode 139. 单词拆分 题目链接&#xff1a;139. 单词拆分 - 力扣&#xff08;LeetCode&#xff09; 这道题使用完全背包来实现&#xff0c;我们首先考虑字符串是否可以由字符串列表组成&#xff0c;因此dp数组大小为n 1 &#xff0c;其意义是&#xff0c;在n个位置时是否能…

在 CentOS 7 上安装 MySQL 8

在 CentOS 7 上安装 MySQL 8 步骤 1: 添加 MySQL Yum 存储库 首先&#xff0c;我们需要添加 MySQL Yum 存储库。打开终端并执行以下命令&#xff1a; sudo yum install -y https://repo.mysql.com/mysql80-community-release-el7-3.noarch.rpm步骤 2: 导入 MySQL GPG 公钥 …

wangeditor实时预览

<template><div><!--挂载富文本编辑器--><div style"width: 45%;float: left;margin-left: 2%"><p>编辑内容</p><div id"editor" style"height: 100%"></div></div><div style"w…

20世纪的葡萄酒有哪些创新?

葡萄酒是用酵母发酵的&#xff0c;直到20世纪中叶&#xff0c;这一过程都依赖于自然产生的酵母。这些发酵的结果往往不一致&#xff0c;而且由于发酵时间长&#xff0c;容易腐败。 酿酒业最重要的进步之一是在20世纪50、60年代引进了地中海的纯发酵菌种酿酒酵母&#xff0c;俗称…

计算机基础知识60

MySQL分组 # 概念&#xff1a;分组是按照某个指定的条件将单个单个的个体分成一个个整体 # MySQL分组的关键字&#xff1a;group by # 分组一般配合聚合函数使用&#xff1a; sum max min avg count 基本的语法格式: group by 字段名 [having 条件表达式] # 单独使用 group by关…

滴滴打车app出现系统异常,已过12小时,部分功能仍未完全恢复

据多地用户反馈&#xff0c;滴滴出行APP无法使用。11月27日深夜&#xff0c;上海、北京、广州等多地滴滴用户反馈&#xff0c;滴滴出行APP无法使用&#xff0c;地图无法加载。 不少网约车司机反映&#xff0c;“滴滴出行”系统出现故障&#xff0c;导致无法接单、定位混乱等情况…

相关性分析和作图

相关的类型 1. Pearson、Spearman和Kendall相关 Pearson 积差相关系数衡量了两个定量变量之间的线性相关程度。&#xff08;连续&#xff09; Spearman等级相关系数则衡量分级定序变量之间的相关程度。&#xff08;分类&#xff09; Kendall’s Tau 相关系数也是一种非参数的…

MySQL实现高可用方案-MHA安装及配置

MySQL高可用性解决方案Master High Availability (MHA) 是一种在 MySQL 故障转移环境中实现快速故障转移和数据保护的开源软件。MHA 能在 MySQL 主节点发生故障时&#xff0c;自动将备节点提升为主节点&#xff0c;并且不会中断正在进行的 SQL 操作。 需求&#xff1a;主从配置…

业务建模工具BPMN

目录 一、什么是BPMN 二、业务流程梳理的重要作用 三、BPMN的全图 四、BPMN的组成 1.BPMN的基本元素&#xff08;2.0&#xff09; 1.1 流对象&#xff08;Flow Objects&#xff09; 1.2 数据&#xff08;Data&#xff09; 1.3 连接对象&#xff08;Connecting Objects&a…

M3VSNET:无监督多度量多视图立体视觉网络(2021年)

M3VSNET&#xff1a;无监督多度量多视图立体视觉网络&#xff08;2021年&#xff09; 摘要1 引言2 相关工作3 实现方法3.1 网络架构 B. Huang, H. Yi, C. Huang, Y. He, J. Liu and X. Liu, “M3VSNET: Unsupervised Multi-Metric Multi-View Stereo Network,” 2021 IEEE Inte…

智能优化算法应用:基于混合蛙跳算法无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于混合蛙跳算法无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于混合蛙跳算法无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.混合蛙跳算法4.实验参数设定5.算法结果6.参考…

2021年12月 Scratch图形化(四级)真题解析#中国电子学会#全国青少年软件编程等级考试

Scratch等级考试(1~4级)全部真题・点这里 一、单选题(共15题,每题2分,共30分) 第1题 下图两个积木的值分别是? A:false true B:false false C:true true D:true false 答案:A 第2题 小猫和小狗是非常好的朋友,他们发明了一种加密方法:用两位数字代表字母。…

qInstallMessageHandler的学习

背景&#xff1a;需要做一个日志系统。 把信息重定向到txt文件中。 参考&#xff1a; QT 调试信息如何输出到文件&#xff08;qDebug/qWarning/qCritical/qFatal&#xff09;-CSDN博客 Qt 之 qInstallMessageHandler&#xff08;重定向至文件&#xff09;-CSDN博客 demo…