【问题标题】:GCC bug in range-based for statement基于范围的 for 语句中的 GCC 错误
【发布时间】:2014-11-19 23:20:59
【问题描述】:

我似乎在 GCC 中遇到了一个奇怪的错误。考虑

for (int i = 5 : {1, 2, 3})
     std::cout << i << ", ";

虽然这个无意义的代码是correctly rejected by Clangit compiles and executes fine on all recent GCC versions。奇怪的是根本没有输出,1, 2, 35, 5, 5 都没有产生。

如果我们现在进一步考虑这段代码

int arr[] {1, 2, 3};
for (int i = 5: arr)
    std::cout << i << ", ";

GCC 警告我们that the array is not used,这意味着循环被忽略了。

此代码是否会调用未定义的行为?还是它格式不正确,不需要诊断?或者它只是一个奇怪的错误?
该标准在 [stmt.ranged]/1 中指定将声明替换为以下行:

for-range-declaration = *__begin;

...这样的声明显然是错误的。

编辑:在reporting the bug 之后,Paolo Carlini 已将其修复为 5.0 版。

【问题讨论】:

  • 不错的发现。您是否报告了该错误?
  • @Aleksandar 这可能不是一个错误,而只是未定义行为的结果,见上文。
  • 不是UB,只是语法无效。 GCC 应该拒绝它,除非它有一个旨在处理它的扩展(这是极不可能的)。
  • 故意不接受,请举报。 gcc.gnu.org/bugs - 谢谢!
  • @Columbo 作为旁注,我在使用 g++4.9.2 编译时得到输出 5。在 clang++ 下失败。

标签: c++ c++11 gcc for-loop g++


【解决方案1】:

这两个例子都是无效的语法,需要诊断。

基于范围的 for 语法在 6.5/1 中:

迭代语句

  • ...
  • for ( for-range-declaration : for-range-initializer ) 声明

for-range-declaration

  • attribute-specifier-seqoptdecl-specifier-seq 声明符

这不允许在冒号之前使用初始化器。

【讨论】:

  • 声明器不是由初始化器组成的吗?
  • @remyabel 不,它没有。 init-declarator 可以包含初始化程序。
猜你喜欢
  • 2015-01-11
  • 1970-01-01
  • 1970-01-01
  • 2014-06-12
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多