为什么 ISO c++17 不允许在条件中使用结构化绑定声明?

Kie*_*etz 3 c++ language-lawyer

如果我使用结构化绑定声明作为条件,Clang 会发出警告:

$ cat hello.cc
int main() {
  struct A { int i; operator bool() { return true; } };
  if (auto [i] = A{0}) {
    return i;
  }
  return -1;
}
Run Code Online (Sandbox Code Playgroud)
$ clang++-10 -std=c++17 hello.cc
hello.cc:3:12: warning: ISO C++17 does not permit structured binding declaration in a condition [-Wbinding-in-condition]                                                                                                                                                                               
  if (auto [i] = A{0}) {                                                                                                                                                                                                                                                                               
           ^~~                                                                                                                                                                                                                                                                                         
1 warning generated.
Run Code Online (Sandbox Code Playgroud)

我在dcl.struct.bindstmt.select中没有看到这一点;我在哪里可以看到这是禁止的?

此外:禁止这样做的理由是什么?

Nat*_*ica 6

if 语句的语法是

if constexpr(opt) ( init-statement(opt) condition) statement
Run Code Online (Sandbox Code Playgroud)

正如您所看到的,condition这是必需的,而init-statement是可选的。这意味着if (auto [i] = A{0})thatauto [i] = A{0}是条件,而不是init-statmentcondition定义为

condition:
    expression
    attribute-specifier-seq(opt) decl-specifier-seq declarator brace-or-equal-initializer
Run Code Online (Sandbox Code Playgroud)

并且不允许结构化绑定,因为其语法

attribute-specifier-seq(opt) decl-specifier-seq ref-qualifier(opt) [ identifier-list ] initializer ;
Run Code Online (Sandbox Code Playgroud)

好消息是,您可以通过在 if 语句中添加条件来获得您想要的结果,例如

if (auto [i] = A{0}; i)
Run Code Online (Sandbox Code Playgroud)