一、关于初始化和赋值
1、初始化
在构造函数形参后的冒号与大括号之间的是构造函数初始值列表(即初始化的过程),大括号内的函数体为空。
class TheNum
{
public:
// 通过构造函数来对成员变量进行初始化
TheNum(int x, int y):a(x),b(y){}
private:
int a, b;
};
2、赋值
此时先调用了默认构造函数,再进行赋值
class TheNum
{
public:
// 此时构造函数为赋值的形式
TheNum(int x, int y) {
a = x;
b = y;
}
private:
int a, b;
};
赋值操作相当于:先用默认的值进行初始化,然后再用提供的实参来进行赋值。相比初始化要多一步,因此其效率要低于直接初始化。
3、构造函数的初始值有时必不可少,如下
(1)、当没有进行初始化,而直接进行赋值时的错误如下:
// 在此没有初始化,直接使用赋值,而出现的各种错误。
class TheNum01
{
private:
int a;
const int b;
int &c;
public:
TheNum01(int x)
{
// 以下执行的为赋值操作,而非初始化
a = x; // 正确
b = x; // 错误,不能为const类型变量赋值
c = a; // 错误,c 没有初始化
}
};
(2)、正确的方式是:构造函数的初始值必不可少,如下所示:
class TheNum01
{
private:
int a;
const int b;
int &c;
public:
TheNum01(int x):a(x),b(x),c(x){}
};
【总结】:在很多类中初始化和赋值都可以,但初始化的效率高于赋值。因为前者直接初始化数据成员,后者则先进行初始化,再进行赋值。 如果数据成员是const 、引用,或者属于某种未提供默认构造函数的类类型,则必须通过构造函数初始值列表为这些成员提供初始值,而不能通过赋值来解决。 所以一般会推荐使用构造函数初始值列表来初始化成员变量。
二、成员初始化顺序
1、成员的初始化顺序与构造函数的初始值列表中出现的顺序无关,而是由 定义类时声明的顺序决定。
class TheNum
{
public:
// 先初始化 a,再初始化b
TheNum(int x, int y):b(x),a(y){}
private:
// a 定义,b后被定义
int a, b;
};