C++23中省略参数列表的lambda表达式的有效性

康桓瑋*_*康桓瑋 30 c++ lambda language-lawyer c++23

根据cppreference,gcc 和 clang 最近都完成了P1102R2(“ Down with ()!”)的实现,这意味着我们可以在 C++23 中更简洁地定义 lambda 表达式。

但是我发现它们与某种形式不一致:

auto l = []<auto> noexcept requires true {};
Run Code Online (Sandbox Code Playgroud)

clang 接受这种形式,而 gcc拒绝其语法。

我应该信任哪个编译器?这个 lambda 在 C++23 中是格式正确还是格式错误?

更新:

或许是迫于舆论压力,clang在我举报后的五天内迅速修复了49736

再进一步尝试时,无意中发现gcc也拒绝了以下有效表单,这让我报告了99850,2周后修复。

auto l = []<auto> requires true -> void {};
Run Code Online (Sandbox Code Playgroud)

Bar*_*rry 32

感谢您提醒我此功能是多么无意义。

正确答案是:不,这不是一个格式良好的 lambda。语法在[expr.prim.lambda.general] 中定义:

在此处输入图片说明

在我们的例子中,首先我们有:

[]<auto> noexcept requires true {};
Run Code Online (Sandbox Code Playgroud)
  • []lambda 引入者
  • <auto>匹配<template-parameter-list>,现在我们知道我们是第二种lambda 表达式。所以在语法上,我们需要跟在requires 子句(可选)然后是lambda-declarator然后是复合语句
  • noexcept不匹配requires-clause,所以现在我们正在解析lambda-declarator。一个lambda-declarator可以开始,(parameter-declaration-clause)但我们没有,所以我们只是在寻找lambda-specifiers。我们将noexcept用作noexcept-specifier 的一部分。
  • requires true不适合attribute-specifier-seqtrailing-return-type所以我们都没有,现在我们完成了lambda-specifiers所以我们完成了lambda-declarator。此时,我们正在寻找复合语句。但是我们没有那个,所以这是一个错误。

基本上,有两个地方可以放置一个requires 子句:直接在模板参数之后,或者,如果我们有函数参数,则在函数参数之后的lambda 说明符之后。所以这有效:

[]<auto> requires true noexcept {};
Run Code Online (Sandbox Code Playgroud)

就像这样:

[]<auto>() noexcept requires true {};
Run Code Online (Sandbox Code Playgroud)

就像这样:

[]<auto> requires true () noexcept requires true { };
Run Code Online (Sandbox Code Playgroud)

但不是 OP 中的那个。

另外,不要写这个。

  • 导致当前规则的观察是,requires 子句要么约束显式模板参数列表,要么约束显式函数参数列表。因此,显式的“&lt;&gt;”列表后面可以跟一个requires子句,显式的“()”列表后面可以跟一个requires子句。如果省略显式列表,则无法为其包含 require 子句。 (4认同)
  • 我今天早上为 llvm 提交了一份 [bug 报告](https://bugs.llvm.org/show_bug.cgi?id=49736),但我不能 100% 确定它的格式不正确。谢谢你让我知道我是对的。 (3认同)
  • @Deduplicator:根本不认为值得创建一个复杂的语法来使非空 *lambda-specifiers* 区分两个潜在的 *requires-clause*。 (2认同)