C++中,构造函数为成员变量赋值的方法有两种:构造函数体赋值和初始化列表。
构造函数体赋值是在构造函数里面为成员变量赋值,如:
class Data
{
public:
//构造函数体赋值
Data(int year,int month,int day)
{
_year = year;
_month = month;
_day = day;
}
private:
int _year;
int _month;
int _day;
};
但是,构造函数体赋值其实是先调用成员变量的默认构造函数进行初始化,再在构造函数体内通过赋值运算符重新赋值,相当于执行了“默认构造+赋值”两步。因此,其本质还是赋值,而不是初始化。因为初始化只能初始化一次,而构造函数体内可以多次赋值。
函数体中多次赋值举例:
class A
{
public://结果为最后一次赋值的结果
A(int a = 0)
{
_a = a;
_a += 2;
if (_a > 1)
{
_a *= 10;
}
}
private:
int _a;
};
而初始化列表是在进入构造函数体之前,就会完成对成员变量的初始化。
初始化列表在构造函数的括号后面,以冒号开头,用逗号分隔成员变量,每个成员变量后面跟一个小括号,里面放初始值。
class Data
{
public:
//初始化列表
Data(int year, int month, int day)
:_year(year)
, _month(month)
, _day(day)
{
//里面可以根据需求进行赋值操作
}
private:
int _year;
int _month;
int _day;
};
初始化列表有很重要的作用,当有些构造函数体赋值无法完成给变量赋初值的操作时,就要用到初始化列表。比如const修饰的成员变量、引用成员变量、自定义类型成员。const修饰的成员变量只能初始化,不能赋值修改;引用成员变量在定义时必须指明时谁的别名;自定义类型成员要调用对应的构造函数。而类中的成员变量只是声明,不能直接赋值。
class A
{
public:
A(int a = 10)
:_a(a)
{ }
private:
int _a;
};
class B
{
public:
B(int& ref)
:_aobj(1)
,_ref(ref)
, _n(20)
{}
private:
A _aobj;//写在初始化列表会调用它自己的构造函数;不写,就必须要有对应的默认构造函数
int& _ref;
const int _n;
int x;//内置类型写上会初始化为对应值,不写上就是随机值
};
成员变量在类中声明次序就是其在初始化列表中的初始化顺序,与其在初始化列表中的先后顺序无关。
class A
{
public:
A(int a = 0)
:_a1(a)//后初始化
,_a2(_a1)//先初始化,结果为随机值
{}
private:
int _a2;
int _a1;
};
int main()
{
A aa1(10);
return 0;
}