类的6个默认成员函数
空类class Date {};
中是不是就是为空,答案是否定的。任何空类,都会自动生成6个默认成员函数。
初始化和清理
构造函数: 主要完成初始化工作
析构函数: 主要完成清理工作
拷贝复制
拷贝构造: 使用同类对象初始化创建对象
赋值重载: 主要是把一个对象赋值给另一个对象
取地址重载
主要是普通对象和const
对象取地址,这两个很少会自己实现
构造函数
在刚开始学习C++,未学构造函数时,老师会说为了保持封装性,把成员变量放到private
里,利用放到public
里的成员函数来进行初始化工作。
#include <iostream>
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 1
return 0;
}
但是学习了构造函数后,就不需要使用这么麻烦的步骤进行初始化工作了。
构造函数是一个特殊的成员函数,名字与类名相同,创建类类型对象时由编译器自动调用,保证每个数据成员都有 一个合适的初始值,并且在对象的生命周期内只调用一次。
#include <iostream>
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 1
return 0;
}
特性
构造函数是特殊的成员函数,需要注意的是,构造函数的虽然名称叫构造,但是需要注意的是构造函数的主要任务并不是开空间创建对象,而是初始化对象。
- 函数名与类名相同。
- 无返回值。
- 对象实例化时编译器自动调用对应的构造函数。(对象定义出来,就自动调用,保证了对象一定是被初始化过的)
- 构造函数可以重载。
#include <iostream>
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 <iostream> 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 <iostream> 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_
前缀。
总结
构造函数进行对象的初始化工作,在对象实例化时自动调用,保证实例化对象一定被初始化。
构造函数时默认成员函数,我们不写时编译器会自动生成一份,我们写了的话编译器就不会生成。
对于我们不写时,编译器自动生成的构造函数:
- 对于内置类型的成员变量不处理
- 对于自定义类型的成员变量会调用该变量自己的默认构造函数。