【c++】类和对象(三)构造函数和析构函数

2024-03-26 00:20

本文主要是介绍【c++】类和对象(三)构造函数和析构函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Alt

🔥个人主页:Quitecoder

🔥专栏:c++笔记仓

Alt

朋友们大家好,本篇文章我们带来类和对象重要的部分,构造函数和析构函数

目录

  • 1.类的6个默认成员函数
  • 2.构造函数
    • 2.1构造函数其他特性
  • 3.构析函数
    • 3.1特性:

1.类的6个默认成员函数

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

任何类在什么都不写时,编译器会自动生成以下6个默认成员函数(用户没有显式实现,编译器会生成的成员函数称为默认成员函数)

class Date {};

在这里插入图片描述

2.构造函数

我们看下面这个类

class Date
{
public:void Init(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 d1;d1.Init(2005, 6, 23);d1.Print();Date d2;d2.Init(2024, 3, 25);d2.Print();return 0;
}

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

构造函数是一种特殊的成员函数,它在创建对象时自动调用,其主要目的是初始化对象。在C++中,构造函数具有与其所属类相同的名称,并且没有返回类型。构造函数可以有参数,也可以没有参数,允许通过不同的方式初始化对象的成员变量。如果一个类定义中没有显式地包含任何构造函数编译器会自动生成一个默认构造函数(只在没有其他任何构造函数时)

特性:

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

那么上面所示的代码构造函数如何写呢?如下:

class Date
{
public:Date(){_year = 1;_month = 2;_day = 3;}
private:int _year;int _month;int _day;
};
int main()
{Date d1;	d1.Print();return 0;
}

这种是不带参数的
在这里插入图片描述
在我们进行实例化 Date d1;时,自动调用构造函数完成初始化,我们可以用汇编代码进行查看:
在这里插入图片描述

我们也可以在其中加入带参数的构造函数,实现函数重载

class Date
{
public:Date(){_year = 1;_month = 2;_day = 3;}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;
};

我们知道,构造函数是实例化调用的,那么如何实现带参数构造函数呢?

代码如下:

 Date d1; // 调用无参构造函数Date d2(2005, 1, 1); // 调用带参的构造函数

注意:如果通过无参构造函数创建对象时,对象后面不用跟括号,否则就成了函数声明

在这里插入图片描述

Date d3();

能不能这样定义呢?
在这里插入图片描述
这里编译错误,即这里并不能与函数的声明区分开,所以书写格式严格按照上述方法来写

如果我们将第一个无参格式屏蔽掉呢?

class Date
{
public:/*Date(){_year = 1;_month = 2;_day = 3;}*/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 d1;	d1.Print();return 0;
}

在这里插入图片描述
代码中出现错误的原因在于,为Date类定义了一个接收三个参数的构造函数,但是没有定义默认构造函数(无参数构造函数)。接着,在main函数中,尝试使用无参数的方式构造d1对象:Date d1;。这在类定义中是非法的,因为一旦定义了自己的构造函数(不管有多少参数),C++编译器就不会自动生成默认构造函数

我们这里也可以通过缺省参数来实现:
在这里插入图片描述
在这里插入图片描述

十分好用

2.1构造函数其他特性

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

class Date
{
public:/*// 如果用户显式定义了构造函数,编译器将不再生成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 d1;d1.Print();return 0;
}

在这里插入图片描述
这个默认生成的函数并没有做什么事情

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

C++把类型分成内置类型(基本类型)和自定义类型。内置类型就是语言提供的数据类型,如:int/char…,自定义类型就是我们使用class/struct/union等自己定义的类型,默认生成的构造函数,对内置类型不做处理,自定义类型会去调用它的默认构造函数

我们看下面这串代码:

class A
{
public:A(){cout << "A()" << endl;_a = 0;}
private:int _a;
};class Date
{
public:void Print(){cout << _year << "-" << _month << "-" << _day << endl;}
private://内置类型int _year ;int _month ;int _day;//自定义类型A _aa;
};
int main()
{Date d1;return 0;
}

默认生成的构造函数,对内置类型不做处理,自定义回去调用他的默认构造
在这里插入图片描述
我们发现调用了A的构造
在这里插入图片描述

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

class Date
{
public:void Print(){cout << _year << "-" << _month << "-" << _day << endl;}	
private:		int _year =1;int _month =2;int _day;A _aa;
};

在声明的位置给缺省值在这里还是声明
在这里插入图片描述

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

思考下面代码能否编译成功?

class Date
{
public:Date(){_year = 1900;_month = 1;_day = 1;}Date(int year = 1900, int month = 1, int day = 1){_year = year;_month = month;_day = day;}
private:int _year;int _month;int _day;
};
// 以下测试函数能通过编译吗?
int main()
{Date d1;return 0;
}

答案是不可以

Date类中定义了两个构造函数,看起来目的是提供一个默认构造函数和一个带默认参数值的构造函数。然而,这里的设计存在冲突,因为两个构造函数都可以作为默认构造函数,这导致了一个重定义的问题

在C++中,如果构造函数的所有参数都有默认值,它就可以被视为无参数调用时的候选构造函数,也就是说,它可以被当作默认构造函数。因此,这个类设计在逻辑上等同于提供了两个默认构造函数,这在C++中是不允许的,会导致编译错误

问题在于,当尝试创建一个不传递任何参数的Date对象(如Dated1;),编译器将无法确定应该调用哪个构造函数,因为两个构造函数都满足调用条件

3.构析函数

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

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

3.1特性:

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

写法如下:

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 st1;st1.Push(1);st1.Push(2);return 0;
}

其中:

~Stack(){if (_array){free(_array);_array = NULL;_capacity = 0;_size = 0;}}

为析构函数,我们定义一个栈,如果不写析构函数,则会发生内存泄漏

在这里插入图片描述

c语言中,我们主动调用Destroy函数

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

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类生成的默认析构函数注意:创建哪个类的对象则调用该类的析构函数,销毁那个类的对象则调用该类的析构函数

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

本节内容到此结束!感谢大家观看!!!

这篇关于【c++】类和对象(三)构造函数和析构函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/846731

相关文章

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

Python函数作用域与闭包举例深度解析

《Python函数作用域与闭包举例深度解析》Python函数的作用域规则和闭包是编程中的关键概念,它们决定了变量的访问和生命周期,:本文主要介绍Python函数作用域与闭包的相关资料,文中通过代码... 目录1. 基础作用域访问示例1:访问全局变量示例2:访问外层函数变量2. 闭包基础示例3:简单闭包示例4

JavaScript对象转数组的三种方法实现

《JavaScript对象转数组的三种方法实现》本文介绍了在JavaScript中将对象转换为数组的三种实用方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友... 目录方法1:使用Object.keys()和Array.map()方法2:使用Object.entr

Python中isinstance()函数原理解释及详细用法示例

《Python中isinstance()函数原理解释及详细用法示例》isinstance()是Python内置的一个非常有用的函数,用于检查一个对象是否属于指定的类型或类型元组中的某一个类型,它是Py... 目录python中isinstance()函数原理解释及详细用法指南一、isinstance()函数

python中的高阶函数示例详解

《python中的高阶函数示例详解》在Python中,高阶函数是指接受函数作为参数或返回函数作为结果的函数,下面:本文主要介绍python中高阶函数的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录1.定义2.map函数3.filter函数4.reduce函数5.sorted函数6.自定义高阶函数

Python中的sort方法、sorted函数与lambda表达式及用法详解

《Python中的sort方法、sorted函数与lambda表达式及用法详解》文章对比了Python中list.sort()与sorted()函数的区别,指出sort()原地排序返回None,sor... 目录1. sort()方法1.1 sort()方法1.2 基本语法和参数A. reverse参数B.

C++读写word文档(.docx)DuckX库的使用详解

《C++读写word文档(.docx)DuckX库的使用详解》DuckX是C++库,用于创建/编辑.docx文件,支持读取文档、添加段落/片段、编辑表格,解决中文乱码需更改编码方案,进阶功能含文本替换... 目录一、基本用法1. 读取文档3. 添加段落4. 添加片段3. 编辑表格二、进阶用法1. 文本替换2

使用MapStruct实现Java对象映射的示例代码

《使用MapStruct实现Java对象映射的示例代码》本文主要介绍了使用MapStruct实现Java对象映射的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、什么是 MapStruct?二、实战演练:三步集成 MapStruct第一步:添加 Mave

C++中处理文本数据char与string的终极对比指南

《C++中处理文本数据char与string的终极对比指南》在C++编程中char和string是两种用于处理字符数据的类型,但它们在使用方式和功能上有显著的不同,:本文主要介绍C++中处理文本数... 目录1. 基本定义与本质2. 内存管理3. 操作与功能4. 性能特点5. 使用场景6. 相互转换核心区别