3.拷贝构造函数
知识点:
拷贝构造函数其实是构造函数的一个重载形式
参数只有一个且必须是类类型对象的引用,使用传值方式编译器直接报错,因为会引发无穷递归调用(附:因为当自定义类型传值调用时要先调用拷贝构造来进行传参时的拷贝、然而在拷贝构造中的参数类型假如我们不用引用,那么就又在拷贝构造处进行了自定义类型的传值调用,这样就会导致无限循环)无穷递归展示:
当我们用引用接收(用指针也行),就能解决这个问题
一般我们会在拷贝构造中的参数加上const防止被改变
同样当我们自己不写拷贝构造时,编译器会自动生成
对内置类型进行浅拷贝、值拷贝(完成字节序的值拷贝, 直接将数据拷贝过去)
对自定义类型会调用他们自己的拷贝构造,过程可能是浅拷贝 / 深拷贝 (针对申请的空间)
拷贝构造函数典型调用场景:
使用已存在对象创建新对象(拷贝构造)
函数参数类型为类类型对象(引用)
函数返回值类型为类类型对象(引用)
细节:
深拷贝和浅拷贝的区别: 深拷贝是针对申请的空间的,对一个动态申请的空间进行拷贝时,不能像内置类型那样直接完成字节序的拷贝(浅拷贝),如果这样会导致拷贝后,两个对象指向同一个申请的空间,当一边改变时会导致另一边受影响 / 他们会重复的进行析构函数释放同一块空间而报错,所以说此时就需要我们自己去写一个拷贝构造(深拷贝,具体看下图,就和构造函数差不多不过要有参数且必须是类对象的引用类型的,然后内部就再申请一块空间给*this,然后将数据拷贝的到新空间中即可),来防止这样的错误发生。
c++规定对内置类型来说是直接拷贝(浅拷贝),对自定义类型来说要进行拷贝构造
4.赋值运算符重载
知识点:
4.1运算符重载:
C++为了增强代码的可读性引入了运算符重载,运算符重载是具有特殊函数名的函数,也具有其返回值类型,函数名字以及参数列表。其返回值类型与参数列表与普通的函数类似。
函数名字为:关键字operator+运算符符号。
函数原型:返回值类型 operator操作符(参数列表)
总结来说:运算符重载就是让自定义类型也能和内置类型一样去使用一些操作符(+ - * /)为此去创建一个专属于自定义类型的操作符
如:
此时这个+=就是一个运算符重载,他其实会其调用我们写好的运算符重载函数
细节:
( .* ) ( :: ) (sizeof ) (? :) ( . ) 注意以上5个运算符不能重载
是否要去对重载运算符,取决于对你是否需要、是否有意义
注意对于在写赋值运算符重载时需要把返回值写成 Date & 因为我们需要连续赋值
用于内置类型的运算符时不能改变其含义,例如:内置的整型+,不能改变+的含义
不能通过连接其他符号来创建新的操作符:比如operator@
4.2赋值运算符重载:
知识点:
本质也是运算符重载,但是赋值运算符是一个默认成员函数
参数类型:const T&,传递引用可以提高传参效率
返回值类型:T&,返回引用可以提高返回的效率,有返回值目的是为了支持连续赋值
返回*this :要复合连续赋值的含义
T & operator=(const T&) 其中T是对象名 如 Date& operator=(const Date& d1)
细节:
我们需要在赋值时分清构造函数和运算符重载的区别
构造函数是在一个对象给另一个对象初始化时的赋值操作(Date d3(d1) 等价于 Date d3 = d1)
运算符重载是对两个已经创建好的对象进行时的赋值操作
用户没有显式实现时,编译器会生成一个默认赋值运算符重载,以值的方式逐字节拷贝(和拷贝构造一样,自定义去调用其赋值重载,内置类型进行浅拷贝),所以说对于像Stack这种不全是内置类型成员的对象需要自己去写一个赋值运算符重载
赋值运算符只能重载成类的成员函数不能成重载成全局函数(因为他是默认成员函数,当写在全局时,类就会再默认生成一个,这样就会发生冲突)
练习:
实现cout 、 cin
这是cout 、cin 所在的类分别是ostream 、 istream
ostream& operator<<(ostream& out,const Date& d)//返回值是 out 为了可以连续使用 { //ostream是cout的类对象,也是库iostream定义的 out << d._year << "年" << d._month << "月" << d._day << "天" << endl; //内部就把out当cout使用即可 return out; } istream& operator>>(istream& in, Date& d)//同理要返回 istream& { in >> d._year >> d._month >> d._day;//同理 .... return in; }
注意因为<< >> 他们有两个操作数 (cout << d1;)
而若是 成员函数 的话第一个参数的位置一定是 *this , 而我们需要把第一个参数放成 cout
所有对此我们重载 << >> 时不能写成 成员函数 而是需要写成一个全局函数
并且因为我们需要使用到 成员日期类对象的成员变量 所以我们要使用到一个超纲的知识点 友元函数,加上友元函数后我们就能 直接在全局函数中去使用类中的私有
在类中写: friend ostream& operator<<(ostream& out,const Date& d)
5.const成员
知识点:
将const修饰的“成员函数”称之为const成员函数,const修饰类成员函数,实际修饰该成员函数隐含的this指针,表明在该成员函数中不能对类的任何成员进行修改。
总结来说就是为了改变this指针类型,将this指针的类型改成const Date ...
细节:
我们可以把所有不需要改变成员变量的成员函数全部写成const成员型
向上面这种没对this加const当遇到const类型时就不行了,所以为了防止这种情况我们对不需要改变的尽量都加上const , 这样对于正常的 Date -> const Date 算是范围的缩小 , 而对于const那也能使用了
练习:
实现日期类
6.取地址及const取地址操作符重载
知识点:
这两个是剩下的最后两个默认成员函数,还是比较简单的,就是会默认重载取地址操作符
对此直接通过代码可以更好的展示:
默认生成的:
重载自己的:
本章完。预知后事如何,暂听下回分解。