详解C++类和对象(中(类的6个默认成员函数))

文章目录

  • 写在前面
  • 1. 类的6个默认成员函数
  • 2. 构造函数
    • 2.1 构造函数的引入
    • 2.1 构造函数的特性
  • 3. 析构函数
    • 3.1 析构函数的引入
    • 3.2 析构函数的特性
  • 4. 拷贝构造函数
    • 4.1 拷贝构造函数概念
    • 4.2 拷贝构造函数的特性
    • 4.3 拷贝构造函数典型调用场景
  • 5. 赋值运算符重载
    • 5.1 运算符重载
    • 5.2 赋值运算符重载
  • 6. const成员函数
  • 7. 取地址及const取地址操作符重载

写在前面

这篇文章详细介绍了类的 6 个默认成员函数,它们是构造函数、析构函数、拷贝构造函数、赋值运算符重载、取地址和 const 取地址操作符重载以及const 成员函数。这些成员函数在 C++ 中是默认生成的,默认成员函数在类的设计和实现中起着非常重要的作用,下面我们来一一介绍。

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

如果一个类中什么成员都没有,简称为空类。
空类中真的什么都没有吗?并不是,在 C++ 中,即使一个类中看起来什么都没有,编译器会自动生成以下6个默认成员函数
默认成员函数:用户没有显式实现,编译器会生成的成员函数称为默认成员函数
在这里插入图片描述

2. 构造函数

2.1 构造函数的引入

例如有如下一个类:

#include <iostream>
using namespace std;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, d2;d1.Init(2024, 2, 5);d2.Init(2024, 2, 6);d1.Print();d2.Print();return 0;
}

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

在C++中为了让对象在实例化的时候能够完成初始化,提供了构造函数。构造函数是一种特殊的成员函数,用于在创建对象时对其进行初始化。在 C++ 中,构造函数的名称与类名相同,不返回任何值,甚至没有 void 类型的返回值。构造函数的主要作用是初始化对象的数据成员,确保对象在创建时有 一个合适的初始值。创建类类型对象时由编译器自动调用,并且在对象整个生命周期内只调用一次

2.1 构造函数的特性

  1. 函数名与类名相同。
  2. 无返回值。
  3. 对象实例化时编译器自动调用对应的构造函数。
    在这里插入图片描述
  4. 构造函数可以重载。
    在这里插入图片描述
  5. 如果类中没有显式定义构造函数,则C++编译器会自动生成一个无参的默认构造函数,一旦
    用户显式定义编译器将不再生成。
    在这里插入图片描述
  6. 关于编译器生成的默认成员函数,我们会有疑惑:不实现构造函数的情况下,编译器会
    生成默认的构造函数。但是看起来默认构造函数又没什么用?d对象调用了编译器生成的默
    认构造函数,但是d对象_year/_month/_day,依旧是随机值。也就说在这里编译器生成的
    默认构造函数并没有什么用??

    其实不然,这是因为C++把类型分成内置类型(基本类型)和自定义类型。内置类型就是语言提供的数据类型,如:int/char…,自定义类型就是我们使用class/struct/union等自己定义的类型,看看下面的程序,就会发现编译器生成默认的构造函数会对自定类型成员_t调用的它的默认成员
    函数。
    在这里插入图片描述
    注意:C++11 中针对内置类型成员不初始化的缺陷,又打了补丁,即:内置类型成员变量在
    类中声明时可以给默认值。

    在这里插入图片描述
  7. 无参的构造函数和全缺省的构造函数都称为默认构造函数,并且默认构造函数只能有一个。
    注意:无参构造函数全缺省构造函数、我们没写编译器默认生成的构造函数,都可以认为是默认构造函数即不需要传参就可以调用的构造函数
    在这里插入图片描述
    在实际编程中,如果对象的初始化值不依赖于外部参数,并且没有特殊的初始化需求,编译器生成的默认构造函数就够用。如果对象的初始化值依赖于外部参数,推荐缺省的默认构造函数,因为全缺省的传不传参都可以调用。

3. 析构函数

3.1 析构函数的引入

例如有如下一个类:

#include <iostream>
using namespace std;class Stack
{public:void Init(int capacity = 4){int* tmp = (int*)malloc(sizeof(int) * capacity);if (tmp == nullptr){perror("malloc fail");exit(-1);}_nums = tmp;_capacity = capacity;_top = 0;}void Destroy(){if (_nums){free(_nums);_nums = nullptr;_capacity = _top = 0;}}void push(int x){//检查扩容//..._nums[_top++] = x;}
private:int* _nums;int _top;int _capacity;};
int main()
{Stack st;st.Init();st.push(1);st.push(1);st.push(1);st.Destroy();return 0;
}

对于上面的Stack类,可以通过Destroy公有方法清理对象中的资源(动态申请的空间),但如果每次使用完对象时都调用该方法来主动的释放资源,未免有点麻烦,那能否在对象销毁之前,就自动调用相关函数来清理对象中的资源呢?

在C++中,可以通过析构函数来实现在对象销毁之前自动清理对象中的资源。析构函数是特殊的成员函数,该函数与构造函数功能相反,完成的不是对象的销毁,局部对象销毁工作是由编译器完成的。而对象在销毁时会编译器会自动调用析构函数,完成对象中资源的清理工作

3.2 析构函数的特性

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

  2. 无参数无返回值类型。

  3. 对象生命周期结束时,C++编译系统系统自动调用析构函数。
    在这里插入图片描述

  4. 一个类只能有一个析构函数。若未显式定义,系统会自动生成默认的析构函数。注意:析构函数不能重载。

  5. 关于编译器自动生成的析构函数,是否会完成一些事情呢?下面的程序我们会看到,编译器
    生成的默认析构函数,对自定类型成员调用它的析构函数。
    在这里插入图片描述

#include <iostream>
using namespace std;
class B
{
public:B(int b = 0){_b = b;cout << "B(int b = 0)" << endl;}~B(){cout << "~B()" << endl;}
private:int _b;
};class A
{
public:private:int _a;B _bb;
};int main()
{A a;return 0;
}

上面程序运行结束后输出:~B()。
而在main方法中根本没有直接创建B类的对象,为什么最后会调用B类的析构函数?
因为:main函数中创建了A类的对象a,而a中包含2个成员变量,其中_a是内置类型成员,销毁时不需要资源清理,最后系统直接将其内存回收即可;而_bb是B类对象,所以在a销毁时,要将其内部包含的B类的_bb对象销毁,所以要调用B类的析构函数。但是:main函数中不能直接调用B类的析构函数,实际要释放的是A类对象,所以编译器会调用A类的析构函
数,而A没有显式提供,则编译器会给A类生成一个默认的析构函数,目的是在其内部调用B类的析构函数,即当A对象销毁时,要保证其内部每个自定义对象都可以正确销毁,main函数中并没有直接调用B类析构函数,而是显式调用编译器为A类生成的默认析构函数。
注意:创建哪个类的对象则调用该类的构造函数,销毁那个类的对象则调用该类的析构函数。

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

4. 拷贝构造函数

4.1 拷贝构造函数概念

拷贝构造函数是C++中的一种特殊成员函数,它通常用于在对象创建时,使用一个现有对象的内容来初始化新对象,从而实现对象的拷贝。
拷贝构造函数:只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),在用已存在的类类型对象创建新对象时由编译器自动调用
语法形式如下:

class Date
{
public://默认构造函数Date(int year = 1900, int month = 1, int day = 1){_year = year;_month = month;_day = day;}//拷贝构造函数,使用同类类型的对象初始化Date(const Date& d){_year = d._year;_month = d._month;_day = d._day;}
private:int _year;int _month;int _day;
};int main()
{Date d1(2024, 2, 6);Date d2(d1);//使用 d1 拷贝构造 d2return 0;
}

4.2 拷贝构造函数的特性

  1. 拷贝构造函数是构造函数的一个重载形式。
    在这里插入图片描述
    需要注意的是:显示的写了拷贝构造以后,编译器就不会生成拷贝构造函数了,同时默认构造函数,编译器不会生成了,因为拷贝构造函数也是构造函数。
    在这里插入图片描述

  2. 拷贝构造函数的参数只有一个必须是类类型对象的引用,使用传值方式编译器直接报错,因为会引发无穷递归调用
    在这里插入图片描述

  3. 若未显式定义,编译器会生成默认的拷贝构造函数。 默认的拷贝构造函数对象按内存存储按
    字节序完成拷贝,这种拷贝叫做浅拷贝,或者值拷贝。
    在这里插入图片描述

  4. 编译器生成的默认拷贝构造函数已经可以完成字节序的值拷贝了,还需要自己显式实现吗?
    当然像上面这种类是没必要的。那么下面的类呢?

#include <iostream>
using namespace std;class Stack
{public://构造函数Stack(int capacity = 4){int* tmp = (int*)malloc(sizeof(int) * capacity);if (tmp == nullptr){perror("malloc fail");exit(-1);}_nums = tmp;_capacity = capacity;_top = 0;}//析构函数~Stack(){cout << "~Stack()" << endl;if (_nums){free(_nums);_nums = nullptr;_capacity = _top = 0;}}private:int* _nums;int _top;int _capacity;
};
int main()
{Stack st1;Stack st2(st1);//st1拷贝构造st2return 0;
}

当我们运行上面的程序时,发现代码崩溃,下面我们来分析一下,为什么代码会崩溃呢?
在这里插入图片描述
因此我们得出结论:类中如果没有涉及资源申请时,拷贝构造函数是否写都可以;一旦涉及到资源申请时,则拷贝构造函数是一定要写的,否则就是浅拷贝。
在这里插入图片描述

4.3 拷贝构造函数典型调用场景

现有如下一个Date类:

#include <iostream>
using namespace std;
class Date
{
public:Date(int year = 1900, int month = 1, int day = 1){_year = year;_month = month;_day = day;}Date(const Date& d){cout << "Date(const Date& d)" << endl;_year = d._year;_month = d._month;_day = d._day;}private:int _year;int _month;int _day;
};
  1. 使用已存在对象创建新对象
int main()
{Date d1;Date d2(d1);//使用 d1 拷贝构造 d2Date d3 = d2;//使用 d2 拷贝构造 d3return 0;
}

运行结果如下:
在这里插入图片描述
2. 函数参数类型为类类型对象

void func(Date d)
{cout << "void func(Date d)" << endl;
}
int main()
{Date d1;func(d1);return 0;
}

运行结果如下:
在这里插入图片描述
3. 函数返回值类型为类类型对象
在这里插入图片描述
结论:为了提高程序效率,一般对象传参时,尽量使用引用类型,返回时根据实际场景,能用引用尽量使用引用。

5. 赋值运算符重载

5.1 运算符重载

C++为了增强代码的可读性引入了运算符重载运算符重载是具有特殊函数名的函数,也具有其返回值类型,函数名字以及参数列表,其返回值类型与参数列表与普通的函数类似。有了运算符重载以后,可以根据需求使得自定义类型的对象也可以像内置类型一样使用操作符。
函数名字为:关键字operator后面接需要重载的运算符符号
函数原型:返回值类型 operator操作符(参数列表)
关于运算符重载,有一下几点需要注意:

  1. 不能通过连接其他符号来创建新的操作符:比如operator@ 。只能重载现有的操作符,比如 operator+,operator+=等。
  2. 重载操作符必须有一个类类型参数。
  3. 用于内置类型的运算符,其含义不能改变,例如:内置的整型+,不能改变其含义。
  4. 作为类成员函数重载时,其形参看起来比操作数数目少1,因为成员函数的第一个参数为隐藏的this。
  5. .* :: sizeof ?: . 注意以上5个运算符不能重载。

举个例子:比如想比较两个日期是否相等。
之前我们想比较两个日期的大小需要写一个比较大小的函数,现在有了运算符重载,我们可以重载运算符 ’ ==’ 使得Date类的对象可以像自定义类型一样使用运算符’ ==’ 来判断两个日期是否相等。

#include <iostream>
using namespace std;
class Date
{
public:Date(int year = 1900, int month = 1, int day = 1){_year = year;_month = month;_day = day;}Date(const Date& d){_year = d._year;_month = d._month;_day = d._day;}
//private:int _year;int _month;int _day;
};
//运算符 == 重载,重载成全局的
bool operator==(const Date& d1, const Date& d2)
{return d1._year == d2._year&& d1._month == d2._month&& d1._day == d2._day;
}
int main()
{Date d1(2024, 2, 6);Date d2(2024, 2, 7);cout << (d1 == d2) << endl;//这里d1 和 d2可以像内置类型一样使用操作符return 0;
}

代码运行结果:
在这里插入图片描述
上面将运算符重载成全局的有个很大的问题(上面能运行是因为将成员变量的访问限定符改成了public),就是运算符重载成全局的就需要成员变量是公有的,那么问题来了,封装性如何保证?
这里其实可以用我们后面学习的友元来解决,或者干脆重载成成员函数
在这里插入图片描述
因此正确的写法为:

class Date
{
public:Date(int year = 1900, int month = 1, int day = 1){_year = year;_month = month;_day = day;}Date(const Date& d){_year = d._year;_month = d._month;_day = d._day;}// 运算符 == 重载,重载成成员函数// bool operator==(Date* this, const Date& d)// 这里需要注意的是,左操作数是this,指向调用函数的对象bool operator==(const Date& d){return _year == d._year&& _month == d._month&& _day == d._day;}
private:int _year;int _month;int _day;
};
int main()
{Date d1(2024, 2, 6);Date d2(2024, 2, 7);cout << (d1 == d2) << endl;//这里d1 和 d2可以像内置类型一样使用操作符//上面那样写是为了提高代码的可读性,我们也可以像下面这样显示的调用cout << (d1.operator==(d2)) << endl;//bool operator==(Date* this, const Date& d2)return 0;
}

总结:运算符函数可以定义为类的成员函数或全局函数。如果运算符函数是类的成员函数,它将自动获得一个隐含的 this 指针,用于访问调用对象的成员;如果是全局函数,则需要在参数列表中显式地传递所有操作数。

5.2 赋值运算符重载

  1. 赋值运算符重载是指重载类中的赋值运算符(=),使得用户能够对自定义类型的对象进行赋值操作。通过赋值运算符重载,可以实现类对象之间的拷贝。
    语法格式为:
    参数类型:const T&,传递引用可以提高传参效率。
    返回值类型:T&,返回引用可以提高返回的效率,有返回值目的是为了支持连续赋值。
    检测是否自己给自己赋值
    返回*this :要复合连续赋值的含义。
//T是类型名
T& operator=(const T& 变量名) 
{// 执行赋值操作// 返回 *this
}
  1. 赋值运算符只能重载成类的成员函数不能重载成全局函数。
    在这里插入图片描述
    正确做法如下:

在这里插入图片描述在这里插入图片描述
3. 用户没有显式实现时,编译器会生成一个默认赋值运算符重载,以值的方式逐字节拷贝。注意:内置类型成员变量是直接赋值的,而自定义类型成员变量需要调用对应类的赋值运算符重载完成赋值。
在这里插入图片描述
既然编译器生成的默认赋值运算符重载函数已经可以完成字节序的值拷贝了,还需要自己实现吗?当然像日期类这样的类是没必要的。像前面我们介绍拷贝构造函数那里的Stack类一样,我们发现程序就会崩溃掉。
原因如下:
在这里插入图片描述
总结:如果类中未涉及到资源管理(动态申请空间),赋值运算符是否实现都可以;一旦涉及到资源管理赋值运算符则必须要实现。

6. const成员函数

使用const修饰的“成员函数”称之为const成员函数,const修饰类成员函数,实际修饰该成员函数隐含的this指针,表明在该成员函数中不能对类的任何成员进行修改。并且在函数声明和定义中都需要加上 const 关键字。
语法如下:

返回类型 函数名() const 
{// 函数体
}

在这里插入图片描述
经过上面的介绍,我们来思考如下几个问题:

  1. const对象可以调用非const成员函数吗?
  2. 非const对象可以调用const成员函数吗?
  3. const成员函数内可以调用其它的非const成员函数吗?
  4. 非const成员函数内可以调用其它的const成员函数吗?

在这里插入图片描述

7. 取地址及const取地址操作符重载

这两个默认成员函数一般不用重新定义 ,编译器默认会生成。

class Date
{
public:Date* operator&(){return this;}const Date* operator&() const{return this;}
private:int _year;int _month;int _day;
};

这两个运算符一般不需要重载,使用编译器生成的默认取地址的重载即可,只有特殊情况,才需
要重载,比如想让别人获取到指定的内容

至此,本片文章就结束了,若本篇内容对您有所帮助,请三连点赞,关注,收藏支持下。

创作不易,白嫖不好,各位的支持和认可,就是我创作的最大动力,我们下篇文章见!

如果本篇博客有任何错误,请批评指教,不胜感激 !!!
在这里插入图片描述

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

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

相关文章

智慧城市:打造低碳未来,引领城市数字化转型新篇章

在“万物皆可数字化”的新时代浪潮下&#xff0c;智慧城市作为未来城市发展的先锋方向&#xff0c;正在以前所未有的速度和规模重塑我们的城市面貌。 智慧城市不仅是一个技术革新的标志&#xff0c;更是城市治理、民生服务等领域全面升级的重要引擎。 一、智慧城市的多元应用领…

MATLAB知识点:矩阵元素的引用

​讲解视频&#xff1a;可以在bilibili搜索《MATLAB教程新手入门篇——数学建模清风主讲》。​ MATLAB教程新手入门篇&#xff08;数学建模清风主讲&#xff0c;适合零基础同学观看&#xff09;_哔哩哔哩_bilibili 节选自第3章 3.3.2 矩阵元素的引用 在讲解矩阵元素的引用之…

Unity 接口、抽象类、具体类对象的配合使用案例

文章目录 示例1&#xff1a;接口&#xff08;Interface&#xff09;示例2&#xff1a;抽象类&#xff08;Abstract Class&#xff09;示例3&#xff1a;结合使用接口与抽象类示例4&#xff1a;多接口实现示例5&#xff1a;抽象类与接口结合 在Unity中使用C#编程时&#xff0c;接…

WebSocket基础详解

文章目录 前言由来简介优缺点适用场景兼容性 API介绍构造函数实例方法send()close() 实例属性ws.readyState&#xff08;只读&#xff09;ws.bufferedAmount&#xff08;只读&#xff09;ws.binaryTypeextensions&#xff08;只读&#xff09;protocol&#xff08;只读&#xf…

算法笔记刷题日记——3.简单入门模拟 3.2 查找元素

刷题日记 3.2 查找元素 B1041 B1004 B1028 B1032 A1011 A1006 A1036 错题记录 B1028 人口普查 某城镇进行人口普查&#xff0c;得到了全体居民的生日。现请你写个程序&#xff0c;找出镇上最年长和最年轻的人。 这里确保每个输入的日期都是合法的&#xff0c;但不一定是合理的…

蓝桥杯嵌入式学习记录——LCD的使用

目录 一、前言 二、LCD代码的移植 三、LCD代码的调用 一、前言 前一篇文章已经简单记录了一下cubeMX软件的使用和LED的点亮&#xff0c;今天来记录一下LCD的使用。LCD的驱动代码有很多&#xff0c;但实际上在蓝桥杯的比赛中用起来非常简单&#xff0c;因为赛点会提供LCD的驱…

mysq开启慢查询日志,对慢查询进行优化

1.创建实验的环境 创建对应的数据库&#xff0c;然后写脚本向数据库中写入400万条的数据 //创建实验用的数据库 CREATE DATABASE jsschool;//使用当前数据库 USE jsschool;//创建学生表 CREATE TABLE student (sno VARCHAR(20) PRIMARY KEY COMMENT 学生编号,sname VARCHAR(20…

微软为新闻编辑行业推出 AI 辅助项目,记者参加免费课程

2 月 6 日消息&#xff0c;微软当地时间 5 日发布新闻稿宣布与多家新闻机构展开多项基于生成式 AI 的合作。微软表示&#xff0c;其使命是确保新闻编辑室在今年和未来拥有创新。 目前建议企业通过微软官方合作伙伴获取服务&#xff0c;可以合规、稳定地提供企业用户使用ChatGP…

C#委托的前世今生

起因 很多C#初学者&#xff0c;都遇到过这样的问题——线程间操作无效&#xff0c;从不是创建控件的线程访问它。 今天就这个问题&#xff0c;展开分析。 溯源 先说下这个问题产生的根源。 大家都知道&#xff0c;程序运行起来之后&#xff0c;首先会有一个主线程&#xff…

用HTML5 + JavaScript实现下雪效果

用HTML5 JavaScript实现下雪效果 <canvas>是一个可以使用脚本 (通常为JavaScript) 来绘制图形的 HTML 元素。 <canvas> 标签/元素只是图形容器&#xff0c;必须使用脚本来绘制图形。 HTML5 canvas 图形标签基础https://blog.csdn.net/cnds123/article/details/…

搜索与图论(一)(深搜,广搜,树与图的存储遍历,拓扑排序)

一、DFS 往深里搜&#xff0c;搜到叶子结点那里&#xff0c;回溯&#xff0c;到可以继续到叶子结点深搜的位置。 1、回溯一定要恢复现场 2、定义一个与当前递归层数有关的终止条件&#xff08;题目要求的东西&#xff09; 3、每层都用循环判断是否存在可以dfs的路 输出数字…

DDoS攻击激增,分享高效可靠的DDoS防御方案

当下DDoS攻击规模不断突破上限&#xff0c;形成了 "网络威胁格局中令人不安的趋势"。专业数据显示&#xff0c;对比2022年上半年与2023年上半年&#xff0c;所有行业的DDoS攻击频率增加了314%。其中零售、电信和媒体公司遭受的攻击规模最大&#xff0c;三个垂直行业的…