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成员变量、自定义类型成员(并且该类没有默认构造函数时),必须将它们
- 类中包含引用成员变量、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的特性,在声明处赋值。
对于引用类型:
引用类型,引用必须在定义的时候初始化,并且不能重新赋值,所以也要写在初始化列表里面。
- 成员变量在类中声明次序就是其在初始化列表中的初始化顺序,与其在初始化列表中的先后次序无关。