在c++中,初始化一般有两种方式,一种在构造函数内通过赋值初始化,一种是通过初始化列表初始化。
下面通过对两者进行比较,来分析初始化成员列表和构造函数内赋值的优缺点。
一、构造函数内赋值:
class Text {
public:
Text() {
std::cout << "I'm Text Construction" << std::endl;
}
Text(int num):_num(num) {
std::cout << "I'm Text Construction" << "num = " << num << std::endl;
}
private:
int _num;
};
class Workers {
public:
//一、构造函数内通过赋值初始化
//缺点:无法对const修饰的_sex、或者引用类型_name赋值
//如果Text类不存在默认构造函数,也会报错
Workers(){
_age = 30;
_weigth = 120;
//这样写 会调用一次默认构造函数,加一次带参构造函数,因为多调用了一次默认构造函数、效率降低
_text = Text(15);
}
private:
int _age;
int _weigth;
//打开以下注释会报错
//const std::string _sex;
//std::string &_name;
Text _text;
};
缺点:
1、无法对const修饰的_sex、或者引用类型_name赋值
如果Text类不存在默认构造函数,也会报错。
2、通过_text = Text(15);对_text赋值,会调用一次默认构造函数,加一次带参构造函数,因为多调用了一次默认构造函数、造成效率降低。
二、通过初始化列表初始化:
/*需要使用初始化成员列表时的几种情况
*1、当数据成员是引用时
*2、当数据成员被const修饰时(因为const或者引用只能被初始化,不能赋值)
*3、当数据成员是对象时,该对象无默认构造函数,即有带参的构造函数时
*4、子类需要初始化父类的private数据成员时
*/
class F {
public:
F(int &num):_num(num) {
std::cout << "F num:" << _num << std::endl;
}
private:
int _num;
};
class WorkersEx : public F
{
public:
WorkersEx(int &num) : _weigth(120),_name("a"),
_sex("M"), //const
_age(num), //引用
_text(num), //对象 只调用一次带参构造函数
F(num) //父类
{}
private:
int &_age;
int _weigth;
const std::string _sex;
std::string _name;
Text _text;
};
优点:
1、可以对const成员初始化
2、可以对引用类型成员进行初始化
3、对对象进行初始化时,比构造函数内赋值效率更高,因为只调用了一次带参的构造函数,不会调用默认构造函数。而且不需要额外的赋值运算。
需要注意的是:初始化列表初始化的顺序和变量声明的顺序一致,和初始化中的顺序无关。
总结:
推荐使用成员初始化列表,因为其不仅可以满足const和引用数据成员初始化的要求,还可以有效预防初始化带参对象时调用默认构造函数的问题。