【问题标题】:assignment operator on empty inizializer_list空初始化器列表上的赋值运算符
【发布时间】:2014-01-14 18:07:53
【问题描述】:

你能解释一下 STL 容器如何处理带有空初始化列表的赋值运算符吗?

什么时候我会做这样的事情:

vector<int> v;
v = { };

调用的函数是不是

vector& operator= (initializer_list<value_type> il);

但是:

vector& operator= (vector&& x);

另一方面,当我对自己的班级做类似的事情时:

struct A {
    A& operator= (const A&) { return *this; }
    A& operator= (A&&) { return *this; }
    A& operator= (initializer_list<int>) { return *this; }
};

/* ... */

A a;
a = { };

代码在 VS2013 上无法编译,并说:

error C2593: 'operator =' is ambiguous

如果列表不为空,它工作正常,它只是调用带有初始化列表的函数。只有当列表为空时才会出现问题,在向量上它调用右值赋值运算符,在我的类上它给出错误。

这种情况在vector和其他容器中是如何处理的?

【问题讨论】:

  • gccclang 都没有相同的错误,但与 vs 确认错误。
  • g++ 和 clang++ 明确选择initializer_List&lt;int&gt; 重载operator=。我认为这是正确的:根据[over.ics.list],这个重载的转换等级应该是Exact Match。对于引用重载,必须创建一个临时对象,因此引用(可能)不会直接绑定;因此转换序列是用户定义的转换。 (请注意定义 this 的 [over.ics.ref] 部分没有提及 list-init;它可能未指定。)
  • @dyp 这就是我的想法
  • @dyp 所以引用绑定指回8.5.3,它列举了不包括initializer_list的直接绑定情况,所以我们回退到8.5,这将把我们带到8.5.4 for list-initializaton 所以我认为它没有被指定,也许是令人费解的。

标签: c++ visual-studio-2013 initializer-list overload-resolution compiler-bug


【解决方案1】:

这似乎是一个错误 clang(see it live) 和 gcc(see it live) 接受这个程序并选择 std::initializer_list 重载看起来是正确的,因为这是一个完全匹配,这在C++ draft standard 部分部分13.3.3.1.5 列表初始化序列 段落中进行了介绍2 来自示例:

void f(std::initializer_list<int>);
f( {1,2,3} ); // OK: f(initializer_list<int>) identity conversion
f( {’a’,’b’} ); // OK: f(initializer_list<int>) integral promotion
f( {1.0} ); // error: narrowing

我们有一个身份转换,这是一个完全匹配

对于参考重载,我们转到第 5 段,它说(强调我的未来):

否则,如果参数是引用,请参见 13.3.3.1.4。 [注意:本节中的规则将适用于初始化底层临时参考。 ——尾注]

表示创建了临时文件,然后我们可以将规则应用于生成的临时文件。这将是一个用户定义的转换,它比精确匹配更糟糕。

所以这不应该是模棱两可的。

更新

看起来有两个与此相关的活动错误:

【讨论】:

  • 我同意你的结论,但是我认为(正如在 OP 的 cmets 中所讨论的那样),无论是直接还是间接地应用 13.3.3.1.4 和 8.5.3 都不能绑定(他们谈论参数表达式)。相反,我会说 13.3.3.1.5/6 中的注释(关于引用的注释)表明临时是从 {} 创建的(通过 8.5.4,根据 13.3.3.1.5 被列为 UDC /5),然后直接绑定(8.5.3“如果初始化表达式是 [...] 类纯右值”,按照 13.3.3.1.4/1 列为精确匹配)。另见 13.3.3.1.1/3
  • @dyp 我明白你的意思,虽然我认为你的意思是13.3.3.1.5/5 中的注释,其中写着 注意:本节中的规则将适用于初始化底层临时参考。 ——尾注.
  • 编号取决于您使用的草稿;) 在最新的 github 存储库中,他们将 initializer_list 和数组部分分成两段。
  • +1 表示答案的正确性和错误报告的链接。但是您应该从 13.3.3.1.5 中引用更多内容,以明确: 1) 在f({1,2,3}) 示例中,不仅123 的转换都是身份转换,而且转换整个列表也是。在我阅读 13.3.3.1.5/2 的文本之前,这对我来说并不明显。 2) 你需要 13.3.3.1.5/3 来证明为vector 调用构造函数实际上是用户定义的转换。 3) 为了完整起见,13.3.3.2/2 明确表明标准转换始终优于用户定义的转换。
  • 哦,还有 5.17/9,它声明 z = {1,2}; 等同于 z.operator({1,2});。这不符合通常的规则,因为{1,2} 不是表达式。
猜你喜欢
  • 2012-03-11
  • 2014-11-20
  • 1970-01-01
  • 2017-06-30
  • 2011-12-24
  • 1970-01-01
  • 2019-07-29
  • 1970-01-01
相关资源
最近更新 更多