一文搞懂C++构造函数的初始化列表

问题引入

阅读下面代码:

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++是根据初始化列表来知道怎么初始化成员变量的,如果没有显示的写初始化列表,函数会默认生成相应的初始化列表。

  1. 每个成员变量在初始化列表只能=出现一次(初始化只能出现一次)。
  2. 类中包含引用成员变量、const成员变量、自定义类型成员(并且该类没有默认构造函数时),必须将它们放在初始化列表进行初始化。
  3. 成员变量在类中声明次序就是其在初始化列表中的初始化顺序,与其在初始化列表中的先后次序无关。

1~3

  1. 每个成员变量在初始化列表只能出现一次(初始化只能出现一次)。
  2. 类中包含引用成员变量、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的特性,在声明处赋值。

在这里插入图片描述


对于引用类型:
引用类型,引用必须在定义的时候初始化,并且不能重新赋值,所以也要写在初始化列表里面。

在这里插入图片描述

该篇博客讲解的更加详细,详情请参考


  1. 成员变量在类中声明次序就是其在初始化列表中的初始化顺序,与其在初始化列表中的先后次序无关。

猜你喜欢

转载自blog.csdn.net/weixin_45153969/article/details/132686616