当你的类中的成员为自定义类型时,假如这个自定义类型内部没有默认构造函数,那么你当前的类就无法生成默认生成的默认构造函数,但是如果你手动写构造函数,只能写无参或全缺省或自动生成的构造函数,那么我们要如何对一个自定义类型进行传入参数的初始化呢?这时候就要用到初始化列表。
class Date
{
public:
Date(int year, int month, int day)
: _year(year)
, _month(month)
, _day(day)
{}
private:
int _year;
int _month;
int _day;
}初始化列表:以一个冒号开始,接着是一个以逗号分隔的数据成员列表,每个"成员变量"后面跟一个放在括号中的初始值或表达式。
构造函数其实并不能称为类的初始化函数,它更像一个赋初值函数,同一个成员在构造函数中可以多次赋值,而初始化只能初始化一次,使用初始化列表更加符合我们对初始化的定义,列表中任何一个成员只能出现一次。
类中包含以下成员,必须放在初始化列表位置进行初始化:
1.引用成员变量
2.const成员变量
3.自定义类型成员(该类没有默认构造函数)
class A
{
public:
A(int a)
:_a(a)
{}
private:
int _a;
};
class B
{
public:
B(int a, int ref)
:_aobj(a)
,_ref(ref)
,_n(10)
{}
private:
A _aobj; // 没有默认构造函数
int& _ref; // 引用
const int _n; // const
};
1和2之所以一定要使用初始化列表,是因为它们必须在定义的时候就初始化,而且初始化后不能进行修改,在函数体内进行的操作实际是赋值,引用和const修饰的对象不能被赋值,因此一定要使用初始化列表。
成员变量在类中声明次序就是其在初始化列表中的初始化顺序,与其在初始化列表中的先后次序无关
explicit关键字
构造函数不仅可以构造与初始化对象,对于单个参数的构造函数,还具有隐式类型转换的作用。
class Date
{
public:
Date(int year)
:_year(year)
{}
explicit Date(int year)
:_year(year)
{}
private:
int _year;
int _month:
int _day;
};
void TestDate()
{
Date d1(2018);
// 用一个整形变量给日期类型对象赋值
// 实际编译器背后会用2019构造一个无名对象,最后用无名对象给d1对象进行赋值
d1 = 2019;
}用explicit修饰构造函数,将会禁止单参构造函数的隐式类型转换。
版权声明:本文为qq_65786321原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。