为什么标准区分直接列表初始化和复制列表初始化?
Why does the standard differentiate between direct-list-initialization and copy-list-initialization?
我们知道T v(x);
被称为直接初始化,而T v = x;
被称为复制初始化,这意味着它将从x
构造一个临时T
,该将被复制/移动到v
中(这很可能被省略)。
对于列表初始化,标准根据上下文区分两种形式。T v{x};
称为直接列表初始化,而T v = {x};
称为复制列表初始化:
§8.5.4 [dcl.init.list] p1
[...]列表初始化可以在直接初始化或复制初始化中进行 上下文;直接初始化上下文中的列表初始化称为直接列表初始化,复制初始化上下文中的列表初始化称为复制列表初始化。[...]
但是,整个标准中每个标准中只有两个参考文献。对于直接列表初始化,在创建临时文件(如T{x}
(§5.2.3/3
)时会提到它。对于复制列表初始化,它适用于返回语句中的表达式,如return {x};
(§6.6.3/2
)。
现在,以下代码片段呢?
#include <initializer_list>
struct X{
X(X const&) = delete; // no copy
X(X&&) = delete; // no move
X(std::initializer_list<int>){} // only list-init from 'int's
};
int main(){
X x = {42};
}
通常,从X x = expr;
模式来看,我们预计代码无法编译,因为X
的 move 构造函数定义为delete
d。但是,最新版本的Clang和GCC编译了上面的代码,并且在挖掘了一点(并找到上面的引用)之后,这似乎是正确的行为。该标准只定义了整个列表初始化的行为,除了上述几点之外,根本不区分这两种形式。好吧,至少据我所知,无论如何。
所以,再次总结我的问题:
如果它们(显然)做完全相同的事情,将列表初始化拆分为两种形式有什么用?
因为他们不做完全相同的事情。如 13.3.1.7 [over.match.list] 中所述:
在复制列表初始化中,如果选择了显式构造函数,则初始化格式不正确。
简而言之,只能在复制列表初始化上下文中使用隐式转换。
这是显式添加的,以使统一初始化不统一。是的,我知道这听起来有多愚蠢,但请耐心等待。
2008年,N2640发布(PDF),研究了统一初始化的当前状态。它专门研究了直接初始化(T{...}
)和复制初始化(T = {...}
)之间的区别。
总而言之,人们担心explicit
构造函数实际上将变得毫无意义。如果我有一些类型T
,我希望能够从整数构造,但我不想要隐式转换,我将构造函数标记为显式。
然后有人这样做:
T func()
{
return {1};
}
如果没有当前的措辞,这将调用我的explicit
构造函数。那么,如果构造函数变化不大,那么使构造函数explicit
有什么好处呢?
使用当前的措辞,您至少需要直接使用该名称:
T func()
{
return T{1};
}
- 复制列表初始化的隐式转换的等级是多少
- 标准是否使用多余的大括号(例如 T{{{10}}})定义列表初始化?
- 我使用向量来创建类对象列表.初始化向量时如何使用参数调用构造函数?
- C++11 中的混合列表初始化
- 我可以列表初始化 std::vector 并完美转发元素吗?
- 无法在声明时使用初始值设定项列表初始化常量字符*/字符串数组的向量
- C++20 从括号中的值列表初始化聚合,不支持内部数组
- 如何在向量列表初始化时避免对象复制以及如何延长临时的生存期
- 默认参数和空列表初始化
- 如何在列表初始化中放置额外的语句?
- C++列表初始化允许多个用户定义的转换
- 列表初始化是否将原子初始化为零
- 使用可变模板列表初始化数组,并放置new
- 使用整数初始化列表初始化长双精度的向量
- 直接列表初始化的自动规则
- 使用初始化列表初始化unique_ptr的容器,继续
- 如何修复"非聚合无法使用初始值设定项列表初始化" <map>
- 直接列表初始化和复制列表初始化之间的差异
- 为什么我可以在不使用赋值运算符的情况下使用列表初始化普通数组
- C++ - 使用类中的初始值设定项列表初始化动态集