Let’s Make C++ Great Again——6个默认成员函数的引入及构造函数
目录
- 类的6个默认成员函数
-
- 初始化和清理
- 拷贝复制
- 取地址重载
- 构造函数
-
- 特性
- 总结
类的6个默认成员函数
空类class Date {};
中是不是就是为空,答案是否定的。任何空类,都会自动生成6个默认成员函数。
初始化和清理
构造函数: 主要完成初始化工作
析构函数: 主要完成清理工作
拷贝复制
拷贝构造: 使用同类对象初始化创建对象
赋值重载: 主要是把一个对象赋值给另一个对象
取地址重载
主要是普通对象和const
对象取地址,这两个很少会自己实现
构造函数
在刚开始学习C++,未学构造函数时,老师会说为了保持封装性,把成员变量放到private
里,利用放到public
里的成员函数来进行初始化工作。
#include using namespace std;class Date{public:void setDate(int year, int month, int day){this->_year = year;this->_month = month;this->_day = day;}void showDate(){cout << this->_year << ' ' << this->_month << ' ' << this->_day;}private:int _year;int _month;int _day;};int main(){Date d1;d1.setDate(2022, 1, 1);d1.showDate(); // 2022 1 1return 0;}
但是学习了构造函数后,就不需要使用这么麻烦的步骤进行初始化工作了。
构造函数是一个特殊的成员函数,名字与类名相同,创建类类型对象时由编译器自动调用,保证每个数据成员都有 一个合适的初始值,并且在对象的生命周期内只调用一次。
#include using namespace std;class Date{public:Date(int year, int month, int day){this->_year = year;this->_month = month;this->_day = day;}void showDate(){cout << this->_year << ' ' << this->_month << ' ' << this->_day;}private:int _year;int _month;int _day;};int main(){Date d1(2022, 1, 1);d1.showDate(); // 2022 1 1return 0;}
特性
构造函数是特殊的成员函数,需要注意的是,构造函数的虽然名称叫构造,但是需要注意的是构造函数的主要任务并不是开空间创建对象,而是初始化对象。
- 函数名与类名相同。
- 无返回值。
- 对象实例化时编译器自动调用对应的构造函数。(对象定义出来,就自动调用,保证了对象一定是被初始化过的)
- 构造函数可以重载。
#include using namespace std;class Date{public:Date(){;}Date(int year, int month, int day){this->_year = year;this->_month = month;this->_day = day;}void showDate(){cout << this->_year << ' ' << this->_month << ' ' << this->_day;}private:int _year;int _month;int _day;};int main(){Date d1(2022, 1, 1); // 调用有参函数d1.showDate();Date d2; // 调用无参函数// 注意:如果通过无参构造函数创建对象时,对象后面不用跟括号,否则就成了函数声明 // 以下代码的函数:声明了d3函数,该函数无参,返回一个日期类型的对象Date d3();return 0;}
-
如果类中没有显式定义构造函数,则C++编译器会自动生成一个无参的默认构造函数,一旦用户显式定义编译器将不再生成。当对无参的构造函数进行注释
发生:
再对有参的构造函数进行注释
对象可创建:
-
无参的构造函数和全缺省的构造函数都称为默认构造函数,并且默认构造函数只能有一个。注意:无参构造函数、全缺省构造函数、我们没写编译器默认生成的构造函数,都可以认为是默认构造函数。(不用参数就可以调用的构造函数就是默认构造函数)
#include using namespace std;class Date{public:Date(){;}Date(int year = 2022, int month = 1, int day = 1){this->_year = year;this->_month = month;this->_day = day;}void showDate(){cout << this->_year << ' ' << this->_month << ' ' << this->_day;}private:int _year;int _month;int _day;};int main(){Date d2;return 0;}
发生错误:
-
在我们不写构造函数时,编译器会自动生成一个默认的无参构造函数,我当时第一次看到这句话时,很不解,生成的这个默认的无参构造函数有什么用呢?
我们不妨看看下面这个例子:#include using namespace std;class Time{public: Time(int a) { cout << "Time()" << endl; _hour = 0; _minute = 0; _second = 0; }private: int _hour; int _minute; int _second;};class Date{private:// 基本类型(内置类型)int _year;int _month;int _day;// 自定义类型Time _t;};int main(){Date d;return 0;}
发生错误:
原因:当我们去创建这个对象时,会自动调用它的编译器生成的默认构造函数(因为我们没有实现Date类的任何构造函数),Data类属性中有一个自定义类型Time类,Date类默认构造函数会去创建这个Time类对象_t,而此时只能调用Time类的无参构造函数,当我们在Time类中实现了有参构造函数后(此时编译器不再自动生成无参的默认构造函数),就会上面上面的错误。 -
成员变量的命名风格
一般建议成员变量加入_
或m_
前缀。
总结
构造函数进行对象的初始化工作,在对象实例化时自动调用,保证实例化对象一定被初始化。
构造函数时默认成员函数,我们不写时编译器会自动生成一份,我们写了的话编译器就不会生成。
对于我们不写时,编译器自动生成的构造函数:
- 对于内置类型的成员变量不处理
- 对于自定义类型的成员变量会调用该变量自己的默认构造函数。