- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我已经模板化了 gray_code
类,该类旨在存储一些无符号整数,其基础位以格雷码顺序存储。这里是:
template<typename UnsignedInt>
struct gray_code
{
static_assert(std::is_unsigned<UnsignedInt>::value,
"gray code only supports built-in unsigned integers");
// Variable containing the gray code
UnsignedInt value;
// Default constructor
constexpr gray_code()
= default;
// Construction from UnsignedInt
constexpr explicit gray_code(UnsignedInt value):
value( (value >> 1) ^ value )
{}
// Other methods...
};
在一些通用算法中,我写了这样的东西:
template<typename UnsignedInt>
void foo( /* ... */ )
{
gray_code<UnsignedInt> bar{};
// Other stuff...
}
在这段代码中,我希望 bar
初始化为零,因此 bar.value
初始化为零。然而,在与意想不到的错误作斗争之后,似乎 bar.value
被初始化为垃圾(准确地说是 4606858)而不是 0u
。这让我很吃惊,所以我去 cppreference.com 看看上面那行到底应该做什么......
据我所知,T object{};
形式对应于 value initialization .我觉得这句话很有趣:
In all cases, if the empty pair of braces {} is used and T is an aggregate type, aggregate-initialization is performed instead of value-initialization.
但是,gray_code
有一个用户提供的构造函数。因此它不是一个集合,因此 aggregate initialization不执行。 gray_code
没有构造函数采用 std::initializer_list
所以 list initialization也不执行。 gray_code
的值初始化应该遵循通常的 C++14 值初始化规则:
1) If T is a class type with no default constructor or with a user-provided default constructor or with a deleted default constructor, the object is default-initialized.
2) If T is a class type without a user-provided or deleted default constructor (that is, it may be a class with a defaulted default constructor or with an implicitly-defined one) then the object is zero-initialized and then it is default-initialized if it has a non-trivial default constructor.
3) If T is an array type, each element of the array is value-initialized.
4) Otherwise, the object is zero-initialized.
如果我没看错,gray_code
有一个显式默认的(不是用户提供的)默认构造函数,因此 1) 不适用。它有一个默认的默认构造函数,所以 2) 适用:gray_code
是 zero-initialized .默认的默认构造函数似乎满足了普通默认构造函数的所有要求,因此不应发生默认初始化。那么我们来看看gray_code
是如何被零初始化的:
If T is a scalar type, the object's initial value is the integral constant zero implicitly converted to T.
If T is an non-union class type, all base classes and non-static data members are zero-initialized, and all padding is initialized to zero bits. The constructors, if any, are ignored.
If T is a union type, the first non-static named data member is zero-initialized and all padding is initialized to zero bits.
If T is array type, each element is zero-initialized
If T is reference type, nothing is done.
gray_code
是非 union 类类型。因此,它的所有非静态数据成员都应该被初始化,这意味着 value
是零初始化的。 value
满足std::is_unsigned
,因此是一个标量类型,这意味着它应该用“整数常量零隐式转换为T”来初始化。
所以,如果我没看错的话,在上面的函数 foo
中,bar.value
应该总是用 0
初始化并且它永远不应该用垃圾初始化,对吗?
注意:我编译代码时使用的编译器是 MinGW_w4 GCC 4.9.1(POSIX 线程和 dwarf 异常),以防万一。虽然我有时会在我的计算机上得到垃圾,但我从来没有设法通过在线编译器得到零。
更新:似乎是一个 GCC 错误,错误是我的,而不是我的编译器的错误。实际上,在写这个问题时,为了简单起见,我假设
class foo {
foo() = default;
};
和
class foo {
foo();
};
foo::foo() = default;
是等价的。他们不是。这是来自 C++14 标准的引用,第 [dcl.fct.def.default] 部分:
A function is user-provided if it is user-declared and not explicitly defaulted or deleted on its first declaration.
换句话说,当我得到垃圾值时,我的默认默认构造函数确实是用户提供的,因为它没有在第一次声明时显式错误。因此,发生的不是零初始化而是默认初始化。再次感谢@Columbo 指出真正的问题。
最佳答案
So, if I read correctly all of that, in the function
foo
above,bar.value
should always be initialized with0
and it should never be initialized with garbage, am I right?
是的。您的对象是直接列表初始化的。 C++14 的* [dcl.init.list]/3 指定
List-initialization of an object or reference of type
T
is defined as follows:
[… Inapplicable bullet points…]
Otherwise, if
T
is an aggregate, aggregate initialization is performed (8.5.1).Otherwise, if the initializer list has no elements and T is a class type with a default constructor, the object is value-initialized.
[…]
您的类不是聚合,因为它具有用户提供的构造函数,但它确实具有默认构造函数。 [dcl.init]/7:
To value-initialize an object of type
T
means:
if
T
is a (possibly cv-qualified) class type (Clause 9) with either no default constructor (12.1) or a default constructor that is user-provided or deleted, then the object is default-initialized;if T is a (possibly cv-qualified) class type without a user-provided or deleted default constructor, then the object is zero-initialized and the semantic constraints for default-initialization are checked, and if
T
has a non-trivial default constructor, the object is default-initialized;
[dcl.fct.def.default]/4:
A special member function is user-provided if it is user-declared and not explicitly defaulted […] on its first declaration.
所以你的构造函数不是用户提供的,因此对象是零初始化的。 (构造函数没有被调用,因为它微不足道)
最后,如果不清楚,零初始化 T
类型的对象或引用意味着:
if
T
is a scalar type (3.9), the object is initialized to the value obtained by converting the integer literal0
(zero) toT
;if
T
is a (possibly cv-qualified) non-union class type, each non-static data member and each base-class subobject is zero-initialized and padding is initialized to zero bits;[…]
因此要么
你的编译器有问题
…或者您的代码在其他某个时间点触发了未定义的行为。
* 在 C++11 中答案仍然是肯定的,尽管引用的部分不等价。
关于c++ - 值初始化 : default initialization or zero initialization?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26699720/
我正在尝试将 keras.initializers 引入我的网络,following this link : import keras from keras.optimizers import RMS
我正在为程序创建某种前端。为了启动程序,我使用了调用 CreateProcess(),其中接收到一个指向 STARTUPINFO 结构的指针。初始化我曾经做过的结构: STARTUPINFO star
我已经模板化了 gray_code 类,该类旨在存储一些无符号整数,其基础位以格雷码顺序存储。这里是: template struct gray_code { static_assert(st
我已经查看了之前所有与此标题类似的问题,但我找不到解决方案。所有错误都表明我没有初始化 ArrayList。我是否没有像 = new ArrayList 这样初始化 ArrayList? ? impo
当涉及到 Swift 类时,我对必需的初始化器和委托(delegate)的初始化器有点混淆。 正如您在下面的示例代码中所见,NewDog 可以通过两种方式中的一种进行初始化。如您所见,您可以通过在初始
几天来我一直在为一段代码苦苦挣扎。我在运行代码时收到的错误消息是: 错误:数组初始值设定项必须是初始值设定项列表 accountStore(int size = 0):accts(大小){} 这里似乎
我想返回一个数组,因为它是否被覆盖并不重要,我的方法是这样的: double * kryds(double linje_1[], double linje_2[]){ double x = linje
尝试在 C++ 中创建一个简单的 vector 时,出现以下错误: Non-aggregates cannot be initialized with initializer list. 我使用的代码
如何在构造函数中(在堆栈上)存储初始化列表所需的临时状态? 例如,实现这个构造函数…… // configabstraction.h #include class ConfigAbstraction
我正在尝试编写一个 native Node 插件,它枚举 Windows 机器上的所有窗口并将它们的标题数组返回给 JS userland。 但是我被这个错误难住了: C:\Program Files
#include using namespace std; struct TDate { int day, month, year; void Readfromkb() {
我很难弄清楚这段代码为何有效。我不应该收到“数组初始值设定项必须是初始值设定项列表”错误吗? #include class B { public: B() { std::cout << "B C
std::map m = { {"Marc G.", 123}, {"Zulija N.", 456}, {"John D.", 369} }; 在 Xcode 中,我将 C+
为了帮助你明白这一点,我给出了我的代码:(main.cpp),只涉及一个文件。 #include #include using namespace std; class test{ public
这在 VS2018 中有效,但在 2008 中无效,我不确定如何修复它。 #include #include int main() { std::map myMap = {
我有一个类: #include class Object { std::shared_ptr object_ptr; public: Object() {} template
我正在为 POD、STL 和复合类型(如数组)开发小型(漂亮)打印机。在这样做的同时,我也在摆弄初始化列表并遇到以下声明 std::vector arr{ { 10, 11, 12 }, { 20,
我正在使用解析实现模型。 这是我的代码。 import Foundation import UIKit import Parse class User { var objectId : String
我正在观看 Java 内存模型视频演示,作者说与 Lazy Initialization 相比,使用 Static Lazy Initialization 更好,我不清楚他说的是什么想说。 我想接触社
如果您查看 Backbone.js 的源代码,您会看到此模式的多种用途: this.initialize.apply(this, arguments); 例如,这里: var Router =
我是一名优秀的程序员,十分优秀!