1、构造函数
包括有参数和无参数的构造函数,即使不声明无参数的构造函数,编译器默认也会产生一个无参数的构造函数。
在实际使用时,可以采用
TEST t1(1) 的方式显示调用构造函数
和
TEST t2 = 1 的方式隐式调用构造函数,注意这种方式,在构造函数explicit声明后不可用
2、拷贝/复制构造函数
新建1个类对象,并且从另外1个类对象中拷贝内容到新建的对象中,
实际使用时,可以采用
TEST t4(t2) 的方式显示调用构造函数
和
TEST t3 = t2 的方式隐式调用拷贝构造函数,拷贝构造函数explicit声明后不可用
3、赋值函数
已经定义的两个类对象之间,通过=操作符进行类成员复制,默认会生成1个赋值函数,但是默认的赋值函数只会进行成员值的简单复制,无法处理指针内容的内存深拷贝。
赋值函数只有在类对象已经存在时才会被调用,如果类对象不存在,则调用的是拷贝构造函数。
4、支持初始化列表的构造函数
支持通过{}的方式快速初始化类对象,主要用于数组等动态长度输入的初始化,需要借助std::initializer_list来实现动态入参的传入。
实际使用时,可以采用
TEST t6 {1, 2, 3} 的方式显示调用动态初始化列表构造函数
TEST t7 = {1, 2, 3} 的方式隐式调用动态初始化列表构造函数,注意在支持动态初始化列表的构造函数explicit声明后不可用
5、由于编译器会自动生成默认拷贝构造函数和赋值函数,如果不想重写拷贝构造函数和赋值函数,又不允许别人使用编译器生成的缺省函数,可以直接将拷贝构造函数和赋值函数声明为private私有函数
6、对于explicit声明,一般对于单参数的构造函数,建议是强制加上explict来禁止隐式调用,这样可以避免隐式转换带来的难以理解的行为,影响代码的可读性。
示例代码如下:
#include <iostream>
#include <vector>
using namespace std;
class TEST
{
public:
TEST() {
cout << "无参数默认构造函数" <<endl;
}
TEST(int argIn):m_num(argIn) {
cout << "构造函数" <<endl;
}
TEST(const TEST& t):m_num(t.m_num) {
cout << "拷贝构造函数" <<endl;
}
TEST& operator= (const TEST& t) {
m_num = t.m_num;
cout << "赋值函数" <<endl;
return *this;
}
TEST(std::initializer_list<int> listArg) {
for (auto it = listArg.begin(); it != listArg.end(); it++) {
m_vec.push_back(*it);
}
cout << "支持动态初始化列表的构造函数" <<endl;
}
~TEST(){
}
private:
int m_num;
std::vector<int> m_vec;
};
int main()
{
TEST t1(1); // 显示调用构造函数
TEST t2 = 1; // 隐式调用构造函数,构造函数explicit声明后不可用
TEST t3 = t2; // 隐式调用拷贝构造函数,拷贝构造函数explicit声明后不可用
TEST t4(t2); // 显示调用构造函数
TEST t5; // 调用默认无参数构造函数
t5 = t2; // 调用赋值函数
TEST t6 {1, 2, 3}; // 显示调用动态初始化列表构造函数
TEST t7 = {1, 2, 3}; // 隐式调用动态初始化列表构造函数,支持动态初始化列表的构造函数explicit声明后不可用
}
打印输出如下:
构造函数
构造函数
拷贝构造函数
拷贝构造函数
无参数默认构造函数
赋值函数
支持动态初始化列表的构造函数
支持动态初始化列表的构造函数