问题引入
阅读下面代码:
class Time
{
public:
Time(int hour, int min, int sec)
{
_hour = hour;
_min = min;
_sec = sec;
}
private:
int _hour;
int _min;
int _sec;
};
我们在实例化对象后,对象自动调用构造函数来初始化自己的值,完成初始化工作,但是我们并不能称它为成员变量的初始化,因为初始化只能初始化一次,然而在函数体的“初始化”工作,准确的说可以赋值多次,所以把函数体赋值当作初始化,有失偏颇。
因此引入初始化列表
1 初始化列表
语法:以一个冒号开始,接着用逗号分隔数据成员列表,每个“成员变量”后面跟一个放在括号中的初始值或表达式。
具体格式如下:
class Time
{
public:
Time(int hour, int min, int sec)
: _hour(hour)
, _min(min)
, _sec(sec)
{
}
private:
int _hour;
int _min;
int _sec;
};
2 注意事项
对象初始化时,首先会根据初始化列表初始化成员变量(如果有自定义类型,会调用相应的默认构造函数),然后再执行函数体的内容。c++是根据初始化列表来知道怎么初始化成员变量的,如果没有显示的写初始化列表,函数会默认生成相应的初始化列表。
- 每个成员变量在初始化列表只能=出现一次(初始化只能出现一次)。
- 类中包含引用成员变量、const成员变量、自定义类型成员(并且该类没有默认构造函数时),必须将它们放在初始化列表进行初始化。
- 成员变量在类中声明次序就是其在初始化列表中的初始化顺序,与其在初始化列表中的先后次序无关。
1~3
- 每个成员变量在初始化列表只能出现一次(初始化只能出现一次)。
- 类中包含引用成员变量、const成员变量、自定义类型成员(并且该类没有默认构造函数时),必须将它们放在初始化列表进行初始化。
想要初始化自定义类型,首先要保证该类型存在默认构造函数(注意什么才是默认构造函数),如果没有会报错,因为当前类如果没有相应的构造函数,编译器会调用该类的默认构造函数,才能执行。代码如下:
对于自定义类型
class Date
{
public:
// 重点在构造函数,其他不是重点
// 全缺省才是默认构造函数
//Date(int year=0, int month=0, int day=0)
//{
// _year = year;
// _month = month;
// _day = day;
//}
Date& operator=(const Date& d1)
{
_year = d1._year;
_month = d1._month;
_day = d1._day;
return *this;
}
private:
int _year;
int _month;
int _day;
};
class Time
{
public:
Time(int hour=0, int min=0, int sec=0)
: _hour(hour)
, _min(min)
, _sec(sec)
{
//在函数体初始化,比较麻烦,必须保证存在默认构造z函数
Date t1;
_d = t1;
}
private:
int _hour;
int _min;
int _sec;
Date _d;
};
int main()
{
Time t(6, 30, 30);
return 0;
}
如果不显示写Date类的默认构造函数,那么编译器不会初始化_d的值,_d里面是随机值。反而如果把初始化的工作写在初始化列表,即便Date类里没有显示写默认构造函数,编译器也会将_d的自定义类型初始化为0.
class Date
{
public:
Date& operator=(const Date& d1)
{
_year = d1._year;
_month = d1._month;
_day = d1._day;
return *this;
}
private:
int _year;
int _month;
int _day;
};
class Time
{
public:
Time(int hour = 0, int min = 0, int sec = 0)
: _hour(hour)
, _min(min)
, _sec(sec)
,_d()
{
}
private:
int _hour;
int _min;
int _sec;
Date _d;
};
int main()
{
Time t(6, 30, 30);
return 0;
}
对于const类型
const修饰的成员变量为常量,不可以在函数体内赋值,因为常量不可修改。但是可以在列表进行初始化,或者使用C++11的特性,在声明处赋值。
对于引用类型:
引用类型,引用必须在定义的时候初始化,并且不能重新赋值,所以也要写在初始化列表里面。
- 成员变量在类中声明次序就是其在初始化列表中的初始化顺序,与其在初始化列表中的先后次序无关。