目录
前言:
我们在C++中引入初始化列表的原因何在?
对于这个问题,我们先回头想想类在实例化的时候,一定会用到构造函数,那么实例化的过程中类成员变量的定义肯定是在构造函数里面了。问题在于是不是在构造函数内部初始化的时候定义呢?接下来我们就探讨一下。
初始化列表的意义:
构造函数体中的语句只能将其称作为赋初值,而不能称作初始化。因为初始化只能初始化一次,而构造函数体内可以多次赋值。
最为重要的是,一些特殊的类成员,必须放在初始化列表位置进行初始化:
1.引用成员变量
2.const成员变量
3.自定义类型成员(该类没有默认构造函数)
由上面的特殊类成员必须放在初始化列表可知:这些特殊的类成员肯定是在初始化列表里定义的,事实上,在C++中规定其他成员也是在初始化列表里定义的。
初始化列表的构造:
初始化列表:以一个冒号开始,接着是一个以逗号分隔的数据成员列表,每个"成员变量"后面跟一个放在括 号中的初始值或表达式。
下面是一个简单的类的构造函数:
第一种是正常的赋值操作,第二种直接用初始化列表来处理。
第一种:
#include<iostream>
using std::cout;
using std::cin;
using std::endl;
class Date
{
public:
Date(int year=1999,int month=10,int day=10)
{
_year = year;
_month = month;
_day = day;
}
private:
int _year;
int _month;
int _day;
};
第二种:
#include<iostream>
using std::cout;
using std::cin;
using std::endl;
class Date
{
public:
Date(int year=1999,int month=10,int day=10)
:_year(year)
,_month(month)
,_day(day)
{}
private:
int _year;
int _month; //在这个地方进行定义
int _day;
};
注:每个成员变量在初始化列表中只能出现一次(初始化只能初始化一次)
特殊类成员的处理:
以自定义类型成员为例:
我们先来看一段有问题的代码
#include<iostream>
using std::cout;
using std::cin;
using std::endl;
class A
{
public:
A(int a) //这里的A没有默认构造函数
{
_a = a;
}
private:
int _a;
};
class Date
{
public:
Date(int year=1999,int month=10,int day=10)
:_year(year)
,_month(month)
,_day(day) //初始化列表此时没对自定义类型成员初始化,导致该成员只有声明,没有定义。
{}
private:
int _year;
int _month;
int _day;
A _a;
};
如果类成员里包括自定义类型A,且在初始化列表里没有初始化定义,那么程序会自动调用A的默认构造函数,上面的代码中A显然没有默认构造函数,那么此时导致了A只有声明,没有定义,编译报错。
解决这样的问题方法大致有两种,一是修改A的构造函数,使其变为默认构造函数。二是在初始化列表里直接对_a进行初始化。
一般情况下,我们都会采取第二种方法来处理,即把所有的成员都在初始化列表里进行处理,且都整成全缺省形式,例:
#include<iostream>
using std::cout;
using std::cin;
using std::endl;
class A
{
public:
A(int a)
{
_a = a;
}
private:
int _a;
};
class Date
{
public:
Date(int year = 1999, int month = 10, int day = 10, int a = 10)
:_year(year)
, _month(month)
, _day(day)
, _a(a) //在这个地方初始化
{}
private:
int _year;
int _month;
int _day;
A _a;
};
初始化顺序:
成员变量在类中声明次序就是其在初始化列表中的初始化顺序,与其在初始化列表中的先后次序无关,例:
class A
{
public:
A(int a)
{
_a = a;
}
private:
int _a;
};
class Date
{
public:
Date(int year = 1999, int month = 10, int day = 10, int a = 10)
:_year(year) //2
, _month(month) //4
, _day(day) //3
, _a(a) //1
{}
private:
A _a;
int _year;
int _day;
int _month;
};
这里的初始化顺序就是 _a _year _day _month