C++类与对象、类的6个默认成员函数、构造函数、析构函数等的介绍

文章目录

  • 前言
  • 一、类的6个默认成员函数
  • 二、构造函数
    • 1. 概念
    • 2. 特性
      • 1. 无参构造函数
      • 2. 带参构造函数
      • 3. 编译器默认生成的无参构造函数
    • 3. 构造函数的初始化
    • 4. 默认构造函数
  • 三、析构函数
    • 1. 概念
    • 2. 特性
    • 3. 编译器默认生成的析构函数的作用
    • 4. 构造函数的使用
  • 总结


前言

C++类与对象、类的6个默认成员函数、构造函数、析构函数等的介绍。


一、类的6个默认成员函数

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

class Date {};

任何类在什么都不写时,编译器会自动生成以下6个默认成员函数。
在这里插入图片描述

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

二、构造函数

1. 概念

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

2. 特性

构造函数是特殊的成员函数,需要注意的是,构造函数虽然名称叫构造,但是构造函数的主要任务并不是开空间创建对象,而是初始化对象
特征如下:

  1. 函数名与类名相同。
  2. 无返回值。
  3. 对象实例化时编译器自动调用对应的构造函数
  4. 构造函数可以重载。

1. 无参构造函数

  • 无参构造函数的类实例化对象时,会自动调用构造函数,不传参也不需要加括号
#include <iostream>
using namespace std;

class Date
{
public:
	// 构造函数函数名与类名相同,无返回值
	Date()
	{
		cout << "Date()" << endl;
		_year = 2003;
		_month = 10;
		_day = 1;
	}
private:
	int _year;
	int _month;
	int _day;
};


int main()
{
	Date d1;

	return 0;
}

在这里插入图片描述

上述代码是一个无参构造函数,构造函数也可以带参。


2. 带参构造函数

  • 带参构造函数的类在实例化对象时,也会自动调用带参构造函数。
  • 但是,在实例化时,要进行传参。
  • 传参方式为:在实例化的对象后加括号然后传入参数。如:
  • Date d1(1945, 8, 15);
#include <iostream>
using namespace std;

class Date
{
public:
	// 构造函数函数名与类名相同,无返回值
	// 带参构造函数
	Date(int year, int month, int day)
	{
		cout << "Date()" << endl;
		_year = year;
		_month = month;
		_day = day;
	}
private:
	int _year;
	int _month;
	int _day;
};


int main()
{
	Date d2(1945, 8, 15);

	return 0;
}

在这里插入图片描述

带参构造函数在类中定义时,可以采用缺省参数,如下:

	Date(int year = 1949, int month = 10, int day = 1)
	{
		cout << "Date()" << endl;
		_year = year;
		_month = month;
		_day = day;
	}

注意:

  • 无参构造函数的类实例化对象时,不需要加括号,否则编译器会认为是函数声明。如下:

在这里插入图片描述

3. 编译器默认生成的无参构造函数

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

#include <iostream>
using namespace std;

class Date
{
public:
	/*Date(int year, int month, int day)
	{
		cout << "Date()" << endl;
		_year = year;
		_month = month;
		_day = day;
	}*/

	
private:
	int _year;
	int _month;
	int _day;
};


int main()
{
	Date d1;

	return 0;
}
  • 上述代码:
  1. 将Date类中构造函数屏蔽后,代码可以通过编译,因为编译器生成了一个无参的默认构造函数。
  2. 将Date类中构造函数放开,代码编译失败,因为一旦显式定义任何构造函数,编译器将不再生成。

在这里插入图片描述

3. 构造函数的初始化

上述代码我们不难发现使用编译器默认生成的构造函数进行初始化,类的三个成员变量依然是随机值。如下:
在这里插入图片描述

原因如下:
C++把类型分成内置类型(基本类型)和自定义类型。内置类型就是语言提供的数据类型,如:int/char…,自定义类型就是我们使用class/struct/union等自己定义的类型。编译器默认生成的无参构造函数:
1. 对自定义类型会调用他自己的构造函数
2. 对内置类型不做处理(有些编译器会处理,但规定上不处理)。

#include <iostream>
using namespace std;

class Time
{
public:
	Time()
	{
		_hour = 0;
		_minute = 0;
		_second = 0;
	}

private:
	int _hour;
	int _minute;
	int _second;
};
class Date
{
private:
	// 内置类型
	int _year;
	int _month;
	int _day;
	// 自定义类型
	Time time;
};
int main()
{
	Date d1;

	return 0;
}

在这里插入图片描述


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

class Date
{
private:
	// 内置类型
	int _year = 1949;
	int _month = 10;
	int _day = 1;
	// 自定义类型
	Time time;
};
int main()
{
	Date d1;

	return 0;
}

在这里插入图片描述

4. 默认构造函数

无参的构造函数和全缺省的构造函数都称为默认构造函数,并且默认构造函数只能有一个。
注意:无参构造函数全缺省构造函数、我们没写编译器默认生成的构造函数,都可以认为是默认构造函数。

比如:

#include <iostream>
using namespace std;

class Date
{
public:
	Date()
	{
		_year = 1949;
		_month = 10;
		_day = 1;
	}

	Date(int year = 1949, int month = 10, int day = 1)
	{
		_year = year;
		_month = month;;
		_day = day;
	}
private:
	int _year;
	int _month;
	int _day;
};

int main()
{
	Date d1;
	
	return 0;
}
  • 上述代码,一个无参构造函数,一个全缺省带参构造函数,共两个默认构造函数,会报错。
  • 原因如下:

在语法上,上述定义时可以的,但是在调用上会产生歧义,全缺省构造函数不传参时,与无参构造函数调用产生歧义。

三、析构函数

1. 概念

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

2. 特性

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

  1. 析构函数名是在类名前加上字符 ~
  2. 无参数无返回值类型。
  3. 一个类只能有一个析构函数。若未显式定义,系统会自动生成默认的析构函数。注意:析构函数不能重载
  4. 对象生命周期结束时,C++编译系统系统自动调用析构函数
#include <iostream>
using namespace std;

typedef int STDataType;
class Stack
{
public:
	Stack(int capacity = 4)
	{
		_a = (STDataType*)malloc(sizeof(STDataType) * capacity);
		if (_a == nullptr)
		{
			perror("Stack malloc");
			return;
		}

		_top = 0;
		_capacity = capacity;
	}

	void Push(STDataType x)
	{
		//..... 扩容
		_a[_top] = x;
		_top++;

	}


	~Stack()
	{
		cout << "~Stack()" << endl;
		free(_a);
		_a = nullptr;

		_top = 0;
		_capacity = 0;
	}
private:
	STDataType* _a;
	int _top;
	int _capacity;
};
int main()
{
	Stack s;
	s.Push(1);
	s.Push(2);

	return 0;
}

在这里插入图片描述

3. 编译器默认生成的析构函数的作用

编译器生成的默认析构函数,对自定类型成员调用它的析构函数。对内置类型将权限还给操作系统即可。

#include <iostream>
using namespace std;
class Time
{
public:
	~Time()
	{
		_hour = 0;
		_minute = 0;
		_second = 0;
	}
private:
	int _hour;
	int _minute;
	int _second;
};

class Date
{
public:
	Date()
	{
		_year = 1368;
		_month = 1;
		_day = 1;
	}
private:
	int _year;
	int _month;
	int _day;

	Time _t;
};

int main()
{
	Date d1;

	return 0;
}

// 在main方法中根本没有直接创建Time类的对象,为什么最后会调用Time类的析构函数?
// 因为:main方法中创建了Date对象d,而d中包含4个成员变量,其中_year, _month,_day三个是内置类型成员,
// 销毁时不需要资源清理,最后系统直接将其内存回收即可;
// 而_t是Time类对象,所以在d销毁时,要将其内部包含的Time类的_t对象销毁,所以要调用Time类的析构函数。
//但是:main函数类的析构函数,而Date没有显式提供,则编译器会给Date类生成一个默认的析构函数,
// 目的是在其内部调用Time类的析构函数,即当Date对象销毁时,要保证其内部每个自定义对象都可以正确销毁
// main函数中并没有直接调用Time类析构函数,而是显式调用编译器为Date类生成的默认析构函数

在这里插入图片描述

4. 构造函数的使用

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


总结

C++类与对象、类的6个默认成员函数、构造函数、析构函数等的介绍。

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

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

相关文章

Scikit-Learn支持向量机回归

Scikit-Learn支持向量机回归 1、支持向量机回归1.1、最大间隔与SVM的分类1.2、软间隔最大化1.3、支持向量机回归1.4、支持向量机回归的优缺点2、Scikit-Learn支持向量机回归2.1、Scikit-Learn支持向量机回归API2.2、支持向量机回归初体验2.3、支持向量机回归实践(加州房价预测…

PLC通过Profibus协议转Modbus协议网关接LED大屏通讯

一、背景 Modbus协议和Profibus协议是两种常用于工业控制系统的通信协议&#xff0c;它们在自动化领域中起着重要的作用。Modbus是一种串行通信协议&#xff0c;被广泛应用于各种设备之间的通信&#xff0c;如传感器、执行器、PLC等。而Profibus则是一种现场总线通信协议&…

随想录Day63 | 单调栈 42. 接雨水 84.柱状图中最大的矩形

随想录Day63 | 单调栈 42. 接雨水 84.柱状图中最大的矩形 42. 接雨水 题目链接 42 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图&#xff0c;计算按此排列的柱子&#xff0c;下雨之后能接多少雨水。 第一次提交 class Solution { public:int trap(vector<int>…

Harbor本地仓库搭建002_Harbor负载均衡节点搭建_nginx安装配置_harbor安装---分布式云原生部署架构搭建002

负载均衡的机器. 可以看到上面是安装nginx的过程 首先去编辑一下yum仓库地址,配置一下nginx的仓库地址 然后这个是配置的内容 然后在进行安装之前最好yum makecache fast 更新一下缓存,这样安装的时候 会安装最新的包 然后就可以安装nginx yum -y install nginx 然后去

解锁微信客服的潜力:提升客户满意度与忠诚度

随着全球数字化进程的加速&#xff0c;企业如何有效利用数字化工具提升服务质量和客户满意度&#xff0c;成为了企业国际化、数字化出海的关键。在这一大背景下&#xff0c;微信客服以其卓越的功能和广泛的用户基础&#xff0c;成为了企业数字化转型的重要助力。 一、微信客服…

西班牙的人工智能医生

西班牙的人工智能医生 西班牙已将自己定位为欧洲负责任人工智能领域的领导者。然而&#xff0c;透明度的承诺往往落空&#xff0c;公共监督机构一直难以获得对司法和福利系统中部署的算法的有效访问。这使得西班牙成为一种日益增长的趋势的一部分&#xff0c;即政府悄悄地试验预…

fastapi修改docs文档页面favicon.ico图标

如下图&#xff0c;文档页面默认使用的是tiangolo大神的Logo 如果打开的标签比较多&#xff0c;就不好区分了&#xff0c;想要修改这个logo&#xff0c;可以用fastapi-cdn-host一行代码搞定 fastapi_cdn_host.patch_docs(app, favicon_url/static/logo.png) 例如&#xff1a;…

SSM名城养老院管理系统-计算机毕业设计源码03948

目 录 摘要 1 绪论 1.1选题的意义 1.2研究现状 1.3Vue.js 主要功能 1.4ssm框架介绍 2 1.5论文结构与章节安排 3 2 名城养老院管理系统分析 4 2.1 可行性分析 4 2.2 系统流程分析 4 2.2.1数据增加流程 5 2.3.2数据修改流程 5 2.3.3数据删除流程 5 2.3 系统功能分析 5 2.3.…

YOLOv10改进 | 主干篇 | YOLOv10引入FasterNeT替换Backbone

1. FasterNeT介绍 1.1 摘要: 为了设计快速神经网络,许多工作一直致力于减少浮点运算(FLOP)的数量。 然而,我们观察到,FLOP 的减少并不一定会导致延迟的类似程度的减少。 这主要源于每秒浮点运算 (FLOPS) 效率低下。 为了实现更快的网络,我们重新审视流行的算子,并证明…

字节豆包大模型API吞吐、函数调用能力、长上下文能力测试总结

离开模型能力谈API价格都是耍流氓&#xff0c;豆包大模型作为API最便宜的模型之一&#xff0c;最近向个人开发者开放了&#xff0c;花了300元和一些时间对模型的API吞吐、函数调用能力、长上下文能力等进行了深度测试&#xff0c;看看它的能力究竟适合做 AI 应用开发吗&#xf…

Mysql事务传播机制

都知道事务传播机制有七种&#xff0c;但是都是 面试背的&#xff0c;实际应用中从来没注意过。这次同事写的时候没注意就给我留了个坑。 有这样一个情况&#xff0c;事务A里边嵌套了事务B&#xff0c;在事务的传播机制上&#xff0c;同事写成了PROPAGATION_REQUIRES_NEW&#…

数字化校园:打造未来教育新风尚

在21世纪的教育蓝图中&#xff0c;"数字化校园"正逐渐从愿景走向现实&#xff0c;它不仅是科技进步与教育创新深度融合的产物&#xff0c;更是重塑教育生态、引领未来学习风尚的关键力量。随着云计算、大数据、人工智能等前沿技术的蓬勃发展&#xff0c;传统的教育模…

C#使用轻量级深度学习模型进行车牌颜色识别和车牌号识别

看到这个文章时候请注意这个不涉及到车牌检测&#xff0c;这个仅仅是车牌颜色和车牌号识别&#xff0c;如果想涉及到车牌检测可以参考这个博客&#xff1a;[C#]winform部署yolov7CRNN实现车牌颜色识别车牌号检测识别_c# yolo 车牌识别-CSDN博客 【训练源码】 https://github.…

台球灯控计费系统安装教程,佳易王桌球房计费系统的安装方法教程

台球灯控计费系统安装教程&#xff0c;佳易王桌球房计费系统的安装方法教程 一、软件操作教程 以下软件操作教程以&#xff0c;佳易王台球计时计费管理软件为例说明 软件文件下载可以点击最下方官网卡片——软件下载——试用版软件下载 1、点击计时开灯&#xff0c;相应的灯…

cad怎么转成pdf文件?方法很简单!

cad怎么转成pdf文件&#xff1f;在数字化时代&#xff0c;CAD图纸的转换与共享已成为日常工作中的常态。无论是建筑设计师、工程师还是学生&#xff0c;都可能遇到需要将CAD文件转换为PDF格式的需求。本文将为您推荐三款高效的CAD转PDF软件&#xff0c;让您轻松实现文件格式的转…

Mask R-CNN

Mask R-CNN 是基于 Faster R-CNN 的改进版本&#xff0c;用于实例分割任务&#xff0c;即在物体检测的基础上进一步为每个目标生成像素级的分割掩码。以下是 Mask R-CNN 的主要改进思路及其关键技术点&#xff1a; 1. 引入分割分支 在 Faster R-CNN 的基础上&#xff0c;Mask…

cms XAMPP搭建帝国cms示例(用于代码审计)

网上大部分都是小皮因为是一键很省事&#xff0c;但本人一直用的xampp所以若有人也是用xampp搭建可以看此篇文章 这里示例为 帝国CMS -v7.5 xampp搭建过程中如果本机存在mysql服务则需要先将mysql服务停止在start xampp的mysql服务 任务管理器----->服务----->找到mys…

这四个有意思的工具,很香

提醒英雄 提醒英雄应用是一款能够帮助用户彻底解决健忘症的应用程序。该应用创建的事项会完全同步到通知中心&#xff0c;并且持续保持在锁屏界面上&#xff0c;只要打开手机&#xff0c;用户就会看到之前设置的提醒事项。这种设计确保了用户在任何时候都能及时收到提醒&#…

软件设计不是CRUD(23):在流式数据处理系统中进行业务抽象落地——详细编码

&#xff08;接上文《软件设计不是CRUD&#xff08;22&#xff09;&#xff1a;在流式数据处理系统中进行业务抽象落地——设计思考》&#xff09; 4、详细设计 项目开发初期&#xff0c;有两种测速雷达和对应的摄像头需要接入&#xff0c;分别是STC500型测速雷达和TTS400型测…

怎么通俗理解概率论中的c r(cramer rao 克拉默拉奥)不等式?

还是推一下比较好记 视频链接 【数理统计学重要定理证明&#xff1a;C-R不等式——无偏估计的方差下界-哔哩哔哩】 https://b23.tv/4gk1AvU 【数理统计学重要定理证明&#xff1a;C-R不等式——无偏估计的方差下界-哔哩哔哩】