一、初始化列表概念
初始化列表是指在类的构造函数后加上冒号 : 及一系列参数
class Date {
private:
int _year;
int _month;
int _day;
public:
Date(int year, int month, int day)
:_year(year)
,_month(month)
,_day(day)
{
_year = year;
_month = month;
_day = day;
}
};
二、初始化列表的作用
对于构造函数的函数体中的内容,并不是初始化,而是初始化后的赋值修改。
事实上,成员变量的定义是在实例化对象时,而成员变量的初始化是在初始化列表时。任何对象的初始化都要先经过初始化列表,即使我们没有显式定义初始化列表,也会经过初始化列表,不过此时的成员变量会被初始化为随机值。我们在成员变量声明时可以设置缺省值,该缺省值其实就是给初始化列表使用的:在没有显式定义初始化列表时,如果没有设置缺省值,那么成员变量被初始化为随机值;如果设置了缺省值,那么成员变量将会被初始化为该缺省值。
当我们没有显式定义初始化列表时,对于内置类型会将其初始化为随机值,对于自定义类型会调用其默认构造函数。
既然成员变量可以通过构造函数的函数体来赋值修改,那初始化列表存在的意义是什么呢?
初始化列表的存在,主要是为了解决三种成员变量的初始化问题:
1.常变量(const修饰的成员变量)
常变量在定义时必须初始化,因此引入初始化列表来对其进行初始化。因为构造函数中的“初始化”是在变量被初始化为随机值后的赋值修改,并非真正的初始化。
class Date {
private:
int _year;
int _month;
int _day;
const int _a;
public:
Date(int year, int month, int day, int a)
:_year(year)
, _month(month)
, _day(day)
, _a(a)
{}
};
2.引用变量
引用变量在定义时必须初始化,因此需要引入初始化列表对其进行初始化。
class Date {
private:
int _year;
int _month;
int _day;
int& _a;
public:
Date(int year, int month, int day, int& a)
:_year(year)
, _month(month)
, _day(day)
, _a(a)
{}
};
3.没有默认构造函数的自定义类型变量
以Time类对象_t为例,如果时间类中没有默认构造函数,那么Date类中在处理_t时无法找到合适默认构造函数调用。因此引入初始化列表来对没有默认构造函数的自定义类型进行初始化。
class Time {
private:
int _hour;
int _mintute;
int _second;
public:
Time(int hour, int mintute, int second)
:_hour(hour)
, _mintute(mintute)
, _second(second)
{}
};
class Date {
private:
int _year;
int _month;
int _day;
Time _t;
public:
Date(int year, int month, int day, int hour, int mintute, int second)
:_year(year)
, _month(month)
, _day(day)
,_t(hour,mintute,second)
{}
};
三、成员变量声明时的缺省值
成员变量声明时设置的缺省值是给初始化列表使用的。如果没有显式定义初始化列表,那么成员变量在经过初始化列表时就会初始化为缺省值(无论是否显式定义初始化列表,成员变量的初始化都会经过初始化列表)。
因此缺省值也可以这样设置:
class A {
private:
int* p1 = nullptr;
int* p2 = (int*)malloc(sizeof(int) * 10);
};