- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我想知道编译器如何处理 lambda 函数而不是常规函数。即使排除捕获列表,正如我认为的那样,它的行为似乎也略有不同。
例如,我在上一篇文章中使用的 Trying to pass a constexpr lambda and use it to explicitly specify returning type ,我使用了一个 constexpr lambda 并将其作为常规函数参数传递。我引用了部分答案。
Parameters to constexpr functions are not themselves constexpr objects - so you cannot use them in constant expressions.
template <typename Lambda_T>
constexpr static auto foo(Lambda_T l) {
return std::array<event, (l())>{};
}
// Compiles with GCC (C++17), though ill-formed (according to the answer of my last post)
不过,引起我注意的是,这确实编译通过了 constexpr lambda
,但没有编译通过了 constexpr 常规(全局)函数。是什么导致了这种差异?常规函数和 lambda 函数之间的编译器行为是否存在其他差异?
编辑:Lambda 实现示例
foo([](){ return 4; }); // C++17 Lambda's are implicitly constexpr
在这种情况下,lambda 基本上是值的包装器。
编辑:全局函数
无论何时传递一个全局函数,编译器都会提示——与使用 lambda 相反——这个函数,无论它是否被定义为 constexpr,都不能在常量条件下使用:
prog.cc:8:20: error: 'l' is not a constant expression
最佳答案
从您的代码段中删除一些无关的东西,我们得到
template<typename T>
constexpr void foo(T t)
{
constexpr int i = t();
}
constexpr int f() { return 42; }
auto l = []{ return 42; }
备注:
您正在尝试将 t()
用作 foo
中的 constexpr
。 foo
实际上可以是一个普通函数,并且仍然表现相同。
lambda 不是函数。它是一个带有 operator()
的匿名结构。
struct L { constexpr int operator()() const { return 42; } };
T
在 foo(l)
调用中被推断为等同于 L
的类型。
T
在 foo(f)
调用中被推导为 int(*)()
。
在这两种情况下,t
不是 foo
中的constexpr。
来自 [expr.const]
An expression
e
is a core constant expression unless the evaluation ofe
, following the rules of the abstract machine, would evaluate one of the following expressions: [...]
an lvalue-to-rvalue conversion unless [...]
a non-volatile glvalue that refers to a non-volatile object defined with constexpr, or that refers to a non-mutable subobject of such an object, or
a non-volatile glvalue of literal type that refers to a non-volatile object whose lifetime began within the evaluation of
e
;
通过 int(*)()
调用需要左值到右值的转换。 t
不是用 constexpr
定义的对象,它的生命周期也不是在 t()
的计算中开始的。因此 t()
不是 foo(f)
中的 constexpr
。
调用 operator()
不需要左值到右值的转换。由于没有成员访问权限,因此它只是一个 constexpr
函数调用。因此 t()
是 foo(l)
中的 constexpr
。
有one more step从核心常量表达式到常量表达式,但对于此处的讨论并不重要。
关于c++ - 编译器如何以不同于常规函数的方式对待 lambda?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54387501/
C语言sscanf()函数:从字符串中读取指定格式的数据 头文件: ?
最近,我有一个关于工作预评估的问题,即使查询了每个功能的工作原理,我也不知道如何解决。这是一个伪代码。 下面是一个名为foo()的函数,该函数将被传递一个值并返回一个值。如果将以下值传递给foo函数,
CStr 函数 返回表达式,该表达式已被转换为 String 子类型的 Variant。 CStr(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CSng 函数 返回表达式,该表达式已被转换为 Single 子类型的 Variant。 CSng(expression) expression 参数是任意有效的表达式。 说明 通常,可
CreateObject 函数 创建并返回对 Automation 对象的引用。 CreateObject(servername.typename [, location]) 参数 serv
Cos 函数 返回某个角的余弦值。 Cos(number) number 参数可以是任何将某个角表示为弧度的有效数值表达式。 说明 Cos 函数取某个角并返回直角三角形两边的比值。此比值是
CLng 函数 返回表达式,此表达式已被转换为 Long 子类型的 Variant。 CLng(expression) expression 参数是任意有效的表达式。 说明 通常,您可以使
CInt 函数 返回表达式,此表达式已被转换为 Integer 子类型的 Variant。 CInt(expression) expression 参数是任意有效的表达式。 说明 通常,可
Chr 函数 返回与指定的 ANSI 字符代码相对应的字符。 Chr(charcode) charcode 参数是可以标识字符的数字。 说明 从 0 到 31 的数字表示标准的不可打印的
CDbl 函数 返回表达式,此表达式已被转换为 Double 子类型的 Variant。 CDbl(expression) expression 参数是任意有效的表达式。 说明 通常,您可
CDate 函数 返回表达式,此表达式已被转换为 Date 子类型的 Variant。 CDate(date) date 参数是任意有效的日期表达式。 说明 IsDate 函数用于判断 d
CCur 函数 返回表达式,此表达式已被转换为 Currency 子类型的 Variant。 CCur(expression) expression 参数是任意有效的表达式。 说明 通常,
CByte 函数 返回表达式,此表达式已被转换为 Byte 子类型的 Variant。 CByte(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CBool 函数 返回表达式,此表达式已转换为 Boolean 子类型的 Variant。 CBool(expression) expression 是任意有效的表达式。 说明 如果 ex
Atn 函数 返回数值的反正切值。 Atn(number) number 参数可以是任意有效的数值表达式。 说明 Atn 函数计算直角三角形两个边的比值 (number) 并返回对应角的弧
Asc 函数 返回与字符串的第一个字母对应的 ANSI 字符代码。 Asc(string) string 参数是任意有效的字符串表达式。如果 string 参数未包含字符,则将发生运行时错误。
Array 函数 返回包含数组的 Variant。 Array(arglist) arglist 参数是赋给包含在 Variant 中的数组元素的值的列表(用逗号分隔)。如果没有指定此参数,则
Abs 函数 返回数字的绝对值。 Abs(number) number 参数可以是任意有效的数值表达式。如果 number 包含 Null,则返回 Null;如果是未初始化变量,则返回 0。
FormatPercent 函数 返回表达式,此表达式已被格式化为尾随有 % 符号的百分比(乘以 100 )。 FormatPercent(expression[,NumDigitsAfterD
FormatNumber 函数 返回表达式,此表达式已被格式化为数值。 FormatNumber( expression [,NumDigitsAfterDecimal [,Inc
我是一名优秀的程序员,十分优秀!