将类成员变量放在初始化列表中不需要在初始化列表中是否有任何好处?例如:
class Foo
{
public:
Foo() {}
};
class Bar
{
public:
Bar() : _foo() {}
private:
Foo _foo;
};
在这种情况下,编译器是否会执行任何特殊操作?
答案 0 :(得分:9)
在这种情况下,它没有任何区别。
但这样做很有用 如果你有大量的成员,那么列表中有一些但不是所有成员会引起一些混乱。此外,它还强调了初始化的顺序(顺序是由类中声明的顺序定义的,但是在较大的类中可视化此顺序非常有用,其中并非所有成员变量都在彼此旁边声明)。 / p>
注意:如果在内部化程序列表中将它们放入错误的顺序,这通常只是大多数编译器的警告(除非您将警告编译为错误(您应该这样做))。
真正的危险在于具有POD成员和编译器生成的构造函数的类。
class NewFoo
{
int x;
int y;
};
// Version 1:
class Bar1
{
NewFoo f;
};
// Version 2:
class Bar2
{
NewFoo f;
public:
Bar2() // f not in list.
{}
};
// Version 3:
class Bar3
{
NewFoo f;
public:
Bar3()
: f()
{}
};
int main()
{
Bar1 b1a; // x and y not initialized.
Bar1 b1b = Bar1(); // x and y zero initialized.
Bar2 b2a; // x and y not initialized.
Bar2 b2b = Bar2(); // x and y not initialized.
Bar3 b3a; // x and y zero initialized.
Bar3 b3b = Bar3(); // x and y zero initialized.
}
答案 1 :(得分:8)
初始化列表是初始化成员的首选方法 构造函数。这是初始化引用的唯一可能方法 成员和常任成员。
同样通过使用初始化列表,您可以减少 在变量出现之前意外使用变量的可能性 初始化。这是从未有过的更大哲学的一部分 在不初始化的情况下定义变量。
答案 2 :(得分:6)
正在使用的构造函数的初始化程序列表中未提及的成员是默认初始化的。对于Foo
,这意味着将调用默认构造函数。
Bar() : _foo() { }
与Bar
显式默认构造函数之间的差异(或者说Bar() = default
是您的版本没有普通的默认构造函数。因此,如果你完全离开了构造函数,std::is_trivially_constructible<Bar>::value
会有所不同,尽管行为是相同的。
答案 3 :(得分:1)
不,在这种情况下没有什么不同。我认为这样做没有任何好处。