C++ lambda匿名函数
Lambda 表达式完整的格式如下:
[捕获列表] (形参列表) mutable 异常列表-> 返回类型
{
函数体
}
各项的含义:
- 捕获列表:捕获外部变量,捕获的变量可以在函数体中使用,可以省略,即不捕获外部变量。
- 形参列表:和普通函数的形参列表一样。可省略,即无参数列表
- mutable:mutable 关键字,如果有,则表示在函数体中可以修改捕获变量,根据具体需求决定是否需要省略。
- 异常列表:noexcept / throw(...),和普通函数的异常列表一样,可省略,即代表可能抛出任何类型的异常。
- 返回类型:和函数的返回类型一样。可省略,如省略,编译器将自动推导返回类型。
- 函数体:代码实现。可省略,但是没意义。
示例:
void LambdaDemo()
{
int a = 1;
int b = 2;
auto lambda = [a, b](int x, int y)mutable throw() -> bool
{
return a + b > x + y;
};
bool ret = lambda(3, 4);
}
对应的汇编代码如下:
LambdaDemo()::{lambda(int, int)#1}::operator()(int, int):
push rbp
mov rbp, rsp
mov QWORD PTR [rbp-8], rdi
mov DWORD PTR [rbp-12], esi
mov DWORD PTR [rbp-16], edx
mov rax, QWORD PTR [rbp-8]
mov edx, DWORD PTR [rax]
mov rax, QWORD PTR [rbp-8]
mov eax, DWORD PTR [rax+4]
lea ecx, [rdx+rax]
mov edx, DWORD PTR [rbp-12]
mov eax, DWORD PTR [rbp-16]
add eax, edx
cmp ecx, eax
setg al
pop rbp
ret
LambdaDemo():
push rbp
mov rbp, rsp
sub rsp, 32
mov DWORD PTR [rbp-4], 1
mov DWORD PTR [rbp-8], 2
mov eax, DWORD PTR [rbp-4]
mov DWORD PTR [rbp-20], eax
mov eax, DWORD PTR [rbp-8]
mov DWORD PTR [rbp-16], eax
lea rax, [rbp-20]
mov edx, 4
mov esi, 3
mov rdi, rax
call LambdaDemo()::{lambda(int, int)#1}::operator()(int, int)
mov BYTE PTR [rbp-9], al
nop
leave
ret
可见,在调用匿名函数时,直接调用了LambdaDemo()::{lambda(int, int)#1}::operator()方法,这也体现出lambda匿名函数的优势:
1、内联优化:Lambda表达式可以被编译器更好地内联优化。由于lambda通常在定义时使用,编译器可以获取更多的上下文信息,从而进行更有效的优化。(能够在需要使用的时候定义,并且无需跳出当前函数)
2、减少函数调用的开销:相比于传统函数,匿名函数可以减少函数调用的开销,因为它们不需要通过函数指针调用,并且可以避免在调用栈上创建额外的帧。
3、泛型编程:C++14引入了泛型lambda表达式,它允许编写更通用的代码,减少了模板特化的需要,这样不仅可以提高代码的可读性,还可以减少编译器生成的代码量,从而可能提高性能。
PS:函数调用汇编代码:

可以看出,在用函数调用来实现上述lambda匿名函数相同的功能时,因未能获取更多的上下文信息,就需要传入更多的参数来实现该功能,从而影响性能
可使用:https://godbolt.org/ 查看对应代码的汇编
编译器原理
编译器实现 lambda 表达式大致分为一下几个步骤
- 1、创建 lambda 类,实现构造函数,使用 lambda 表达式的函数体重载 operator()(所以 lambda 表达式 也叫匿名函数对象)
- 2、创建 lambda 对象
- 3、通过对象调用 operator()
class lambda_xxxx
{
private:
int a;
int b;
public:
lambda_xxxx(int _a, int _b) :a(_a), b(_b)
{
}
bool operator()(int x, int y) throw()
{
return a + b > x + y;
}
};
void LambdaDemo()
{
int a = 1;
int b = 2;
lambda_xxxx lambda = lambda_xxxx(a, b);
bool ret = lambda.operator()(3, 4);
}
其中,类名 lambda_xxxx 的 xxxx 是为了防止命名冲突加上的。
lambda_xxxx 与 lambda 表达式 的对应关系
- lambda 表达式中的捕获列表,对应 lambda_xxxx 类的private 成员
- lambda 表达式中的形参列表,对应 lambda_xxxx 类成员函数 operator() 的形参列表
- lambda 表达式中的mutable,对应 lambda_xxxx 类 成员函数operator()的常属性 const,即是否是常成员函数
- lambda 表达式中的返回类型,对应 lambda_xxxx 类成员函数 operator() 的返回类型
- lambda 表达式中的函数体,对应 lambda_xxxx 类成员函数 operator() 的函数体
另外,lambda 表达 捕获列表的捕获方式,也影响 对应 lambda_xxxx 类的 private 成员 的类型
- 值捕获:private 成员 的类型与捕获变量的类型一致
- 引用捕获:private 成员 的类型是捕获变量的引用类型
注:
如果该匿名函数未用mutable关键词修饰,如下:
auto func1 = [](){cout << "hello world!" << endl; };
func1();
对应的类:(着重关注operator()的定义,属性)
template<typename T=void>
class TestLambda01
{
public:
TestLambda01() {}
void operator()()const
{
cout << "hello world!" << endl;
}
};
operator()它是const的 哦^_ ^
C++ lambda匿名函数的更多相关文章
- lambda匿名函数透析
lambda匿名函数透析 目录 1 匿名函数的作用... 1 2 匿名函数的格式... 1 3 匿名函数实例代码... 3 1 匿名函数的作用 ...
- lambda 匿名函数
# 普通python函数 def func(a,b,c): return a+b+c print func(1,2,3) # 返回值为6 # lambda匿名函数 f = lambda a,b,c:a ...
- lambda 形参:返回值 lambda 匿名函数 格式:
lambda 匿名函数 格式: lambda 形参:返回值 e.g f = lambda n:n**2 print(f(10))
- Python 进阶 之 lambda 匿名函数
lambda 是个匿名函数,通常用于简单判断或者处理,例如判断一个数的奇偶性,过滤字符串,逻辑运算等等. lambda表达式: >>>lambda x:x*x >>> ...
- xpinyin-函数返回多个值-lambda匿名函数-列表生成式-三元表达式
import xpinyinp=xpinyin.Pinyin() #实例化print(p.get_pinyin('小白','')) 函数返回多个值:1.函数如果返回多个值的话,它会把这几个值放到一个元 ...
- lambda匿名函数和他的小伙伴(处理大量数据的时候用到)
lambda匿名函数 主要是为了解决一些简单的需求而设计的一句话函数 #计算n的n次方 def func(n): return n**n f = lambda n : n ** n 语法: 函数名 = ...
- lambda匿名函数,sorted(),filter(),map(),递归函数
1.lambda匿名函数 为了解决一些简单的需求而设计的一句话函数 #计算n的n次方 def func(n): return n**n print(func(10)) f = lambda n: n* ...
- python基础-4 函数参数引用、lambda 匿名函数、内置函数、处理文件
上节课总结 1.三元运算 name=“name1”if 条件 else “name2” 2.深浅拷贝 数字.字符串 深浅,都一样 2.其他 浅拷贝:只拷贝第一层 深拷贝:不拷贝最后一层 3.set集合 ...
- Python 之父为什么嫌弃 lambda 匿名函数?
Python 支持 lambda 匿名函数,其扩展的 BNF 表示法是lambda_expr ::= "lambda" [parameter_list] ":" ...
- Java中的lambda匿名函数使用
Java中的lambda匿名函数使用 lambda匿名函数的使用是为了满足某些情况下需要临时定义函数,或者事先定义,需要时才使用.在python里面,lambda表达式的表达方式为:lambda 参数 ...
随机推荐
- 如何修改JSONObject 的值
问 题 { "result": { "total": "3", "shops": [ { "shopId&qu ...
- 浅谈processing-java.exe应用程序的使用(与PowerShell的联合)
简单总结一下processing-java.exe的使用,以及和PowerShell结合,如何互相调用和传参. Processing-java 这是 processing-java.exe 的官方说明 ...
- 使用word模板的科研论文编写
编写SCD论文等的时候,可能出现官网的论文模板不够全面.一般我们使用latex作为论文编写模板,格式等都方便控制和编写,而word模板操作起来较为复杂.但是官网有些时候可能找不到latex的模板内容, ...
- 2个月搞定计算机二级C语言——真题(8)解析
1. 前言 本篇我们讲解2个月搞定计算机二级C语言--真题8 2. 程序填空题 2.1 题目要求 2.2 提供的代码 #include <stdio.h> #define N 3 #def ...
- gin解决CORS跨域问题
直接设置跨域参数 新建 cors 文件 package cors import ( "time" "github.com/gin-contrib/cors" & ...
- base64编码与一般的ASCII码和二进制编码有什么不同?base64详解
在密码学实践中,经常会用到Base64编码.比如大名鼎鼎的密码学挑战题"Matasano Crypto Challenges"的第一集合的第一题,就是要求把一个Hex编码的字符串转 ...
- 【前端JSP思考】JSP中#{},${}和%{}的区别
JSP中#{},${}和%{}的区别: #{} #{}:对语句进行预编译,此语句解析的是占位符?,可以防止SQL注入, 比如打印出来的语句 select * from table where id=? ...
- SpringBoot 部署:外置依赖包
目录: 1.前言 2.瘦身前的Jar包 3.解决方案 一.前言 SpringBoot部署起来虽然简单,如果服务器部署在公司内网,速度还行,但是如果部署在公网(阿里云等云服务器上),部署起来实在头疼:编 ...
- 测试工作中用到的Redis命令
由于项目测试的需要,经常需要连接Redis数据库修改某些键值,无奈最近Redis的客户端连接工具使用不了 只有使用命令行来操作了,现总结如下: 1.远程连接Redis redis-cli -h hos ...
- 详细介绍Mybatis的缓存机制
一.缓存机制 1.缓存概述 缓存:缓存就是一块内存空间,保存临时数据 作用:将数据源(数据库或者文件)中的数据读取出来存放到缓存中,再次获取时直接从缓存中获取,可以减少和数据库交互的次数,提升程序的性 ...