calm27 Asked:2020-04-01 18:04:58 +0800 CST2020-04-01 18:04:58 +0800 CST 2020-04-01 18:04:58 +0800 CST C++ 类成员初始化顺序 772 在 C++ 中初始化类成员的顺序是什么?是先调用类构造函数,还是先初始化类成员? class A {} class B { public: B() = default; private: A a; static A a0; int i; static int i0; } c++ 3 个回答 Voted Harry 2020-04-01T18:25:01+08:002020-04-01T18:25:01+08:00 从标准: 非委托构造函数中的初始化按以下顺序完成: ■ (13.1) 首先,并且仅对于最派生类 (1.8) 的构造函数,虚拟基类按照它们在遍历有向非循环时出现的顺序进行初始化图按“从下到上从左到右”的方向,其中“从左到右”表示基类出现在派生类的基说明符列表中的顺序。 ■ (13.2) 然后,直接基类按照它们在基说明符列表中出现的顺序进行初始化(不管内存初始化器的顺序如何)。 ■ (13.3) 接下来,非静态数据成员按照它们在类声明中的声明顺序进行初始化(同样,不考虑 mem 初始化程序的顺序)。 ■ (13.4) 最后,执行构造函数体的复合语句。评论。需要按声明顺序进行初始化,因为它确保基类子对象和成员按其初始化的相反顺序销毁。 那些。在您的 B 类中,将调用构造函数 A,然后调用成员a;如果您提供了它的初始化程序(在构造函数中, type或 type 的默认值),i将在之后(按声明顺序)初始化。aB():i(5){...}int i = 6; 静态成员在使用类之前被初始化一次。 “我想是的”(c)维尼 如果有问题 - 我希望标准中的专家能纠正我(使用正确的术语和对标准的正确引用......) Best Answer Vlad from Moscow 2020-04-01T18:33:00+08:002020-04-01T18:33:00+08:00 首先,类的静态成员将按照在类对象首次创建之前声明它们的顺序进行初始化。也就是说,在调用类构造函数之前。 至于非静态类成员,它们是否在您的示例上下文中初始化取决于类对象的创建方式。 例如,当创建这样的对象时 B b1; 默认情况下会初始化类成员,这对于类B意味着,例如,类成员i实际上不会以任何方式初始化,因为该成员在类定义中没有初始化程序。 然而,有了这样一个类对象的声明B B b2 = B(); 类对象将被值初始化,给定一个默认构造函数,这意味着其标量类成员的零初始化。所以在这种情况下,类成员i将被初始化为 0。 来自 C++17 标准(9.3 Initializers) 11 初始化器为空括号集的对象,即 (),应进行值初始化。 和 8 对 T 类型的对象进行值初始化意味着: (8.2) — 如果 T 是(可能是 cv 限定的)类类型,没有用户提供或删除的默认构造函数,则该对象被零初始化...... 请记住,在声明类的对象时,无论如何都会调用构造函数,但构造函数的主体会在创建对象的成员之后进行控制。为了处理在构造函数主体中以特殊方式初始化的已创建对象成员,可以使用构造函数初始化列表。 例如, struct A { A() : i( 10 ) { std::cout << "i = " << i << '\n'; } int i; }; AnT stands with Russia 2020-04-01T21:39:22+08:002020-04-01T21:39:22+08:00 如果使用构造函数初始化类,则类构造函数负责初始化类的非静态成员。所以你关于“首先”发生什么的问题没有意义。此外,您的类B可以通过值初始化绕过构造函数进行初始化。 非内联静态类成员的初始化顺序取决于它们的定义点。您根本没有提供静态成员的定义,因此不可能谈论它们的初始化顺序。我们只能说这些变量很可能在它们第一次使用之前就被初始化(尽管在某些情况下这也可能是一个问题)。
从标准:
那些。在您的 B 类中,将调用构造函数 A,然后调用成员
a
;如果您提供了它的初始化程序(在构造函数中, type或 type 的默认值),i
将在之后(按声明顺序)初始化。a
B():i(5){...}
int i = 6;
静态成员在使用类之前被初始化一次。
“我想是的”(c)维尼
如果有问题 - 我希望标准中的专家能纠正我(使用正确的术语和对标准的正确引用......)
首先,类的静态成员将按照在类对象首次创建之前声明它们的顺序进行初始化。也就是说,在调用类构造函数之前。
至于非静态类成员,它们是否在您的示例上下文中初始化取决于类对象的创建方式。
例如,当创建这样的对象时
默认情况下会初始化类成员,这对于类
B
意味着,例如,类成员i
实际上不会以任何方式初始化,因为该成员在类定义中没有初始化程序。然而,有了这样一个类对象的声明
B
类对象将被值初始化,给定一个默认构造函数,这意味着其标量类成员的零初始化。所以在这种情况下,类成员
i
将被初始化为 0。来自 C++17 标准(9.3 Initializers)
和
请记住,在声明类的对象时,无论如何都会调用构造函数,但构造函数的主体会在创建对象的成员之后进行控制。为了处理在构造函数主体中以特殊方式初始化的已创建对象成员,可以使用构造函数初始化列表。
例如,
如果使用构造函数初始化类,则类构造函数负责初始化类的非静态成员。所以你关于“首先”发生什么的问题没有意义。此外,您的类
B
可以通过值初始化绕过构造函数进行初始化。非内联静态类成员的初始化顺序取决于它们的定义点。您根本没有提供静态成员的定义,因此不可能谈论它们的初始化顺序。我们只能说这些变量很可能在它们第一次使用之前就被初始化(尽管在某些情况下这也可能是一个问题)。