编译器会自动生成默认构造函数吗?隐式生成的默认构造函数是否执行零初始化?如果您从法律上解析2003年标准的语言,则答案是肯定的,否。但是,这还不是全部,因为与用户定义的默认构造函数不同,从头创建对象时并不总是使用隐式定义的默认构造函数 -还有其他两种情况:无构造和成员明智的值初始化。“无构造”的情况实际上只是一种技术问题,因为它在功能上与调用琐碎的默认构造函数没有什么不同。另一种情况是更加有趣:成员明智值初始化通过使用调用“()” [仿佛明确调用一个没有参数的构造函数]和它绕过什么技术上称为的默认构造函数。相反,它对每个数据成员递归执行值初始化,对于原始数据类型,最终将解析为零初始化。因此,实际上,编译器提供了两个不同的隐式定义的默认构造函数。其中一个确实对原始成员数据执行零初始化,而另一个则不执行。以下是一些示例,说明了如何调用每种类型的构造函数: MyClass a; // default-construction or no construction MyClass b = MyClass(); // member-wise value-initialization和 new MyClass; // default-construction or no construction new MyClass(); // member-wise value-initialization注意:如果确实存在用户声明的默认构造函数,则按成员值初始化将简单地调用它并停止。这是标准对此的详细说明...如果不声明构造函数,则编译器会隐式创建一个默认构造函数[12.1-5]默认构造函数不初始化基本类型[12.1-7]MyClass() {} // implicitly defined constructor如果使用“()”初始化对象,则不会直接调用默认构造函数。相反,它会引发很长的规则序列,称为值初始化 [8.5-7]值初始化的最终效果是,永远不会调用隐式声明的默认构造函数。相反,将调用递归的按成员值初始化,最终将对所有原始成员进行零初始化,并在具有用户声明的构造函数的任何成员上调用默认构造函数[8.5-5]值初始化甚至适用于原始类型-它们将被零初始化。[8.5-5]a = int(); // equivalent to int a=0;对于大多数目的来说,所有这些实际上都是没有实际意义的。类的编写者通常不能假定数据成员在隐式初始化序列中将被清零-因此,如果任何自管理类具有需要初始化的原始数据成员,则它应该定义自己的构造函数。那么什么时候重要呢?在某些情况下,通用代码可能要强制初始化未知类型。值初始化提供了一种方法。请记住,如果用户提供了构造函数,则不会发生隐式零初始化。默认情况下,std :: vector包含的数据是值初始化的。这可以防止内存调试器识别与其他情况下未初始化的内存缓冲区相关的逻辑错误。vector::resize( size_type sz, T c=T() ); // default c is "value-initialized"可以使用值初始化语法对原始类型或“普通数据”(POD)类型结构的整个数组进行零初始化。new int[100]();这篇文章提供了有关标准版本之间变化的更多详细信息,并且还指出了在主要编译器中以不同方式应用标准的情况。
C ++会生成一个默认的构造函数,但前提是您不提供自己的构造函数。该标准并没有说明将数据成员清零。默认情况下,当您第一次构造任何对象时,它们是未定义的。这可能会造成混淆,因为大多数C ++基本类型确实具有默认的“构造函数”,这些默认构造函数会将它们初始化为零(int(),bool(),double(),long()等),但是编译器不会调用他们像对象对象一样初始化POD成员。值得注意的是,STL 确实使用这些构造函数来默认构造保存原始类型的容器的内容。