类的默认成员函数:用户没显示显现,编译器自动生成的成员函数,主要有6种,下面依次介绍:
1.构造函数
1.1 定义:并非开空间创造对象,而是对象实例化时初始化对象 如:替代Stack...Data类中写得Init函数功能。
1.2特点:
1.函数名与类名相同
2.无返回值(规定)
3.构造函数可重载
4.对象实例化时系统会自动调用对应的构造函数
5.若类中没有显示定义构造函数,则C++编译器会自动生成一个无参的默认构造函数,
用户显示定义,编译器将不再生成
1.3 无参构造函数 ,全缺省构造函数 ,我们不写构造时编译器默认生成的构造函数 (三者)均叫做默认构造函数
注:若通过无参构造函数创建对象时,对象后面不用跟括号,要区分这里是函数声明还是实例化对象。
1.4这里编译器默认生成MyQueue的构造函数调用了Stack的构造,完成了两个成员的初始化
2.析构函数
2.1 定义:析构函数与构造函数功能相反,析构函数不是完成对对象本身的销毁,如局部对象是存在栈帧的,函数结束时栈帧销毁,她就释放了,不需要我们管。C++规定对象在销毁时会自动调用析构函数,完成对象中资源的清理释放工作。(析构函数的功能类比Stack中的Destroy功能,而像Data没有Destroy,其实没有资源要释放,严格说Data是不需要析构的)
2.2 特点:
1.析构函数是在类名前加上字符~
2.无参数无返回值
3.一个类只能有一个析构函数。若未显式定义,系统会生成默认的析构函数
4.对象生命周期结束时,系统会自动调用析构函数
5.与构造函数类似,编译器自动生成的析构函数对内置类型成员不处理,自定义成员会调用她的析构函数
6.自定义成员无论什么时候都会自动调用析构函数
7.一个局部域的多个对象,C++规定后定义的先析构
3.拷贝构造函数
3.1若构造函数的第一个参数是自身类型的引用,且任何额外的参数都有默认值,则该构造函数也叫做拷贝构造函数,即拷贝构造是一个特殊的构造函数。小技巧:如一个类实现了析构并释放了资源,那么她就要显示写拷贝构造,否则不需要。
拷贝实质:1.把原对象销毁 2.再创造一块空间 3.再拷贝值
3.2 特点:
1.拷贝构造是构造函数的重载
2.其第一个参数是类类型对象的引用(若传值会引发无穷递归),其他参数须有缺省值
3.自定义类型对象进行拷贝行为必须调用拷贝构造
4.若未显式定义拷贝构造,编译器会自动生成拷贝构造的函数。
自动生成的拷贝构造对内置类型成员会完成浅拷贝。自定义类型成员则会调用它的拷贝构造
5.传值传参都会产生一个临时变量调用拷贝构造。
传值引用返回,返回的是对象的别名,没有产生拷贝(但可能会引发空引用问题)
3.3深拷贝与浅拷贝
浅->一个字符一个字符的拷贝;
深->不仅拷贝对象的中的数据,指向的资源也要拷贝一份;
浅拷贝存在的问题:
1.空间会free两次
2.一个对象修改数据,会影响另一个(深拷贝是又创造了一个空间,所以不会有该问题)
4.赋值运算符重载
4.1运算符重载
~当运算符被用于类类型的对象时,C++允许我们用运算符重载的方式指定新的含义。规定:类类型对象使用运算符时,必须调用相应的运算符重载。
~运算符重载函数的参数个数和该运算符作用的运算对象数量一样多(左边传给第一个参数,右边传给第二个参数)
~若一个重载函数是成员函数,则他的第一个运算对象默认传给隐式的this指针,因此运算符重载作为成员函数时,参数比运算对象少了一个
.* :: sizeof ?: . 以上五个运算符不能重载;
重载运算符不能改变内置类型对象的含义,如不能将+的含义变为-
~前置++ 和 后置++ 运算符重载函数名都是operator++,为了区分规定后置++重载时添加一个int形参;重载<<和>>时,需要重载为全局函数,因为若为成员函数this指针会占用第一个形参位置,与<<的使用习惯不同。
4.2赋值运算符重载
~是一个默认成员函数,用于完成两个已经存在的对象直接拷贝复制,与拷贝构造不同,拷贝构造用于一个对象拷贝初始化给另一个要创建的对象
~特点:
1.她是一个运算符重载,规定必须重载为成员函数,
其参数建议写成const当前类型的引用,可以减少传值传参的拷贝。
2.有返回值,且建议写成当前类型的引用,
引用返回提高效率,有返回值的目的是支持连续赋值,如a=b=c
5.实践:实现日期类
Date.h
Date.cpp
6取地址运算符重载
6.1const成员函数
~是const修饰的成员函数,如void Print() const. const实际修饰的是隐含的this指针,表明在该成员函数中不能对类的任何成员进行修改。
如const修饰成员函数Date类的Print函数,Print隐含的this指针
Date* const this ---> const Date* const this
6.2取地址运算符重载
~编译器自动实现的就够了(若我们不想让别人取到当前类对象的地址,就可以自己实现一份,胡乱返回一个地址)
class Date
{
public:
Date* operator&()
{
return 011213x;
}
};