C++ Lambda 表达式的本质及原理分析
目录
1.引言
2.Lambda 的本质
3.Lambda 的捕获机制的本质
4.捕获方式的实现与底层原理
5.默认捕获的实现原理
6.捕获 this 的机制
7.捕获的限制与注意事项
8.总结
1.引言
C++ 中的 Lambda 表达式是一种匿名函数,最早在 C++11 引入,用于简化函数对象的定义和使用。它以更简洁的语法提供了强大的功能,但其本质和捕获机制背后有许多值得深究的细节。本文将探讨 Lambda 的本质,以及捕获的底层实现与原理。
2.Lambda 的本质
Lambda 是一个语法糖,本质上是由编译器生成的一个匿名类,该类重载了 operator()(即调用运算符)。在使用 Lambda 表达式时,编译器会隐式生成一个这样的类,并在必要时捕获上下文中的变量。
示例与编译器生成的代码对比
#include <iostream>
#include <functional>int main() {int x = 10;auto lambda = [x](int y) { return x + y; };std::cout << lambda(20) << std::endl; // 输出 30return 0;
}
编译器会将上述 Lambda 转换为类似以下的代码:
#include <iostream>
#include <functional>class LambdaClass {int x;
public:LambdaClass(int x) : x(x) {}int operator()(int y) const {return x + y;}
};int main() {int x = 10;LambdaClass lambda(x);std::cout << lambda(20) << std::endl; // 输出 30return0;
}
可以看到,Lambda 实际上是一个具有捕获变量 x 的函数对象。
3.Lambda 的捕获机制的本质
Lambda 的捕获机制允许其在定义时绑定外部作用域中的变量,以便在 Lambda 内部使用。这一机制本质上是通过捕获变量并存储为匿名类的成员变量来实现的。
捕获的两种方式
1)值捕获(capture by value): 捕获外部变量的副本,保存在 Lambda 的内部。
2)引用捕获(capture by reference): 捕获外部变量的引用,Lambda 内部直接访问外部变量。
4.捕获方式的实现与底层原理
1)值捕获的实现 值捕获会在 Lambda 表达式创建时,将捕获的变量拷贝到匿名类的成员变量中。每次调用 Lambda 时,使用的是捕获时的副本。
#include <iostream>int main() {int x = 10;auto lambda = [x]() { std::cout << x << std::endl; };x = 20;lambda(); // 输出 10,而非 20return 0;
}
编译器生成的代码类似于:
class Lambda {int x; // 保存捕获的副本
public:Lambda(int x) : x(x) {}void operator()() const {std::cout << x << std::endl;}
};
这里,x 是一个副本,与原始变量脱离关系。
2)引用捕获的实现 引用捕获则是将外部变量的引用存储为 Lambda 类的成员变量,调用时直接操作原变量。
#include <iostream>int main() {int x = 10;auto lambda = [&x]() { std::cout << x << std::endl; };x = 20;lambda(); // 输出 20return 0;
}
编译器生成的代码类似于:
class Lambda {int& x; // 保存外部变量的引用
public:Lambda(int& x) : x(x) {}void operator()() const {std::cout << x << std::endl;}
};
可以看到,引用捕获直接存储的是外部变量的引用,Lambda 的调用会影响原变量。
5.默认捕获的实现原理
1)默认值捕获 [=]: 使用 [=] 会默认按值捕获外部作用域的所有变量。
int x = 10, y = 20;
auto lambda = [=]() { return x + y; }; // 默认值捕获 x 和 y
等价于:
class Lambda {int x, y;
public:Lambda(int x, int y) : x(x), y(y) {}int operator()() const {return x + y;}
};
2)默认引用捕获 [&]: 使用 [&] 会默认按引用捕获外部作用域的所有变量。
int x = 10, y = 20;
auto lambda = [&]() { return x + y; }; // 默认引用捕获 x 和 y
等价于:
class Lambda {int& x, & y;
public:Lambda(int& x, int& y) : x(x), y(y) {}int operator()() const {return x + y;}
};
6.捕获 this 的机制
捕获 this 时,实际上是按值捕获了 this 指针,使得 Lambda 可以访问当前对象的成员变量。如果捕获 *this,则表示按值捕获整个对象。
示例:捕获 this
#include <iostream>class MyClass {int data = 42;
public:auto createLambda() {return [this]() { std::cout << data << std::endl; };}
};int main() {MyClass obj;auto lambda = obj.createLambda();lambda(); // 输出 42return0;
}
编译器生成的代码类似于:
class Lambda {MyClass* obj; // 捕获 this 指针
public:Lambda(MyClass* obj) : obj(obj) {}void operator()() const {std::cout << obj->data << std::endl;}
};
7.捕获的限制与注意事项
1)不能捕获动态生成的变量: Lambda 只能捕获作用域中已有的变量,不能捕获运行时动态生成的变量。
2)捕获的生命周期: 引用捕获的变量必须保证 Lambda 的生命周期不超过捕获对象。
3)与 mutable 相关的限制: 捕获的变量默认是不可变的(即 const)。如果需要修改捕获的变量,需要显式添加 mutable。
8.总结
1)Lambda 的本质: 是一个匿名类,其捕获的变量存储为类的成员变量,调用时通过重载的 operator() 实现。
2)捕获的本质: 值捕获是将外部变量的副本存储为类成员,引用捕获是将外部变量的引用存储为类成员。
3)注意事项: 使用 Lambda 时,需要特别关注变量的生命周期和捕获方式,以避免未定义行为。
Lambda 表达式在 C++ 中提供了极大的灵活性和简洁性,特别是在需要定义短小的回调函数或处理算法时。理解并熟练使用 Lambda 表达式可以显著提升代码的可读性和效率。
相关文章:
C++ Lambda 表达式的本质及原理分析
目录 1.引言 2.Lambda 的本质 3.Lambda 的捕获机制的本质 4.捕获方式的实现与底层原理 5.默认捕获的实现原理 6.捕获 this 的机制 7.捕获的限制与注意事项 8.总结 1.引言 C 中的 Lambda 表达式是一种匿名函数,最早在 C11 引入,用于简化函数对象的…...
《多线程基础之条件变量》
【条件变量导读】条件变量是多线程中比较灵活而且容易出错的线程同步手段,比如:虚假唤醒、为啥条件变量要和互斥锁结合使用?windows和linux双平台下,初始化、等待条件变量的api一样吗? 本文将分别为您介绍条件变量在w…...
21款炫酷烟花合集
系列专栏 《Python趣味编程》《C/C趣味编程》《HTML趣味编程》《Java趣味编程》 写在前面 Python、C/C、HTML、Java等4种语言实现18款炫酷烟花的代码。 Python Python烟花① 完整代码:Python动漫烟花(完整代码) Python烟花② 完整…...
智能风控 数据分析 groupby、apply、reset_index组合拳
目录 groupby——分组 本例 apply——对每个分组应用一个函数 等价用法 reset_index——重置索引 使用前编辑 注意事项 groupby必须配合聚合函数、 关于agglist 一些groupby试验 1. groupby对象之后。sum(一个列名) 2. groupby对象…...
Python网络自动化运维---用户交互模块
文章目录 目录 文章目录 前言 实验环境准备 一.input函数 代码分段解析 二.getpass模块 前言 在前面的SSH模块章节中,我们都是将提供SSH服务的设备的账户/密码直接写入到python代码中,这样很容易导致账户/密码泄露,而使用Python中的用户交…...
【JVM】调优
目的: 减少minor gc、full gc的次数,也就是减少STW的时间,因为java虚拟机在做后台垃圾收集线程的时候,会停掉其他线程,专门做垃圾收集,这样会影响网站的性能,以及用户的体验。 调优位置&#x…...
软件测试 —— jmeter(2)
软件测试 —— jmeter(2) HTTP默认请求头(元件)元件作用域和取样器作用域HTTP Cookie管理器同步定时器jmeter插件梯度压测线程组(Stepping Thread Group)参数解析总结 Response Times over TimeActive Thre…...
为什么LabVIEW适合软硬件结合的项目?
LabVIEW是一种基于图形化编程的开发平台,广泛应用于软硬件结合的项目中。其强大的硬件接口支持、实时数据采集能力、并行处理能力和直观的用户界面,使得它成为工业控制、仪器仪表、自动化测试等领域中软硬件系统集成的理想选择。LabVIEW的设计哲学强调模…...
【机器学习】自定义数据集 使用tensorflow框架实现逻辑回归并保存模型,然后保存模型后再加载模型进行预测
一、使用tensorflow框架实现逻辑回归 1. 数据部分: 首先自定义了一个简单的数据集,特征 X 是 100 个随机样本,每个样本一个特征,目标值 y 基于线性关系并添加了噪声。tensorflow框架不需要numpy 数组转换为相应的张量࿰…...
.NET Core缓存
目录 缓存的概念 客户端响应缓存 cache-control 服务器端响应缓存 内存缓存(In-memory cache) 用法 GetOrCreateAsync 缓存过期时间策略 缓存的过期时间 解决方法: 两种过期时间策略: 绝对过期时间 滑动过期时间 两…...
GA-CNN-LSTM-Attention、CNN-LSTM-Attention、GA-CNN-LSTM、CNN-LSTM四模型多变量时序预测一键对比
GA-CNN-LSTM-Attention、CNN-LSTM-Attention、GA-CNN-LSTM、CNN-LSTM四模型多变量时序预测一键对比 目录 GA-CNN-LSTM-Attention、CNN-LSTM-Attention、GA-CNN-LSTM、CNN-LSTM四模型多变量时序预测一键对比预测效果基本介绍程序设计参考资料 预测效果 基本介绍 基于GA-CNN-LST…...
git Bash通过SSH key 登录github的详细步骤
1 问题 通过在windows 终端中的通过git登录github 不再是通过密码登录了,需要本地生成一个密钥,配置到gihub中才能使用 2 步骤 (1)首先配置用户名和邮箱 git config --global user.name "用户名"git config --global…...
《企业应用架构模式》笔记
领域逻辑 表模块和数据集一起工作-> 先查询出一个记录集,再根据数据集生成一个(如合同)对象,然后调用合同对象的方法。 这看起来很想service查询出一个对象,但调用的是对象的方法,这看起来像是充血模型…...
深入理解 C 语言函数指针的高级用法:(void (*) (void *)) _IO_funlockfile
深入理解 C 语言函数指针的高级用法 函数指针是 C 语言中极具威力的特性,广泛用于实现回调、动态函数调用以及灵活的程序设计。然而,复杂的函数指针声明常常让即使是有经验的开发者也感到困惑。本文将从函数指针的基本概念出发,逐步解析复杂…...
【JavaSE】图书管理系统
前言:为了巩固之前学习的java知识点,我们用之前学习的java知识点(方法,数组,类和对象,封装,继承,多态,抽象类,接口)来实现一个简单的图书管理系统…...
【C++数论】880. 索引处的解码字符串|2010
本文涉及知识点 数论:质数、最大公约数、菲蜀定理 LeetCode880. 索引处的解码字符串 给定一个编码字符串 s 。请你找出 解码字符串 并将其写入磁带。解码时,从编码字符串中 每次读取一个字符 ,并采取以下步骤: 如果所读的字符是…...
C++/stack_queue
目录 1.stack 1.1stack的介绍 1.2stack的使用 练习题: 1.3stack的模拟实现 2.queue的介绍和使用 2.1queue的介绍 2.2queue的使用 2.3queue的模拟实现 3.priority_queue的介绍和使用 3.1priority_queue的介绍 3.2priority_queue的使用 欢迎 1.stack 1.1stack…...
浅谈APP之历史股票通过echarts绘图
浅谈APP之历史股票通过echarts绘图 需求描述 今天我们需要做一个简单的历史股票收盘价格通过echarts进行绘图,效果如下: 业务实现 代码框架 代码框架如下: . 依赖包下载 我们通过网站下载自己需要的涉及的图标,勾选之后进…...
Ubuntu 20.04 x64下 编译安装ffmpeg
试验的ffmpeg版本 4.1.3 本文使用的config命令 ./configure --prefixhost --enable-shared --disable-static --disable-doc --enable-postproc --enable-gpl --enable-swscale --enable-nonfree --enable-libfdk-aac --enable-decoderh264 --enable-libx265 --enable-libx…...
【橘子Kibana】Kibana的分析能力Analytics简易分析
一、kibana是啥,能干嘛 我们经常会用es来实现一些关于检索,关于分析的业务。但是es本身并没有UI,我们只能通过调用api来完成一些能力。而kibana就是他的一个外置UI,你完全可以这么理解。 当我们进入kibana的主页的时候你可以看到这样的布局。…...
边缘计算医疗风险自查APP开发方案
核心目标:在便携设备(智能手表/家用检测仪)部署轻量化疾病预测模型,实现低延迟、隐私安全的实时健康风险评估。 一、技术架构设计 #mermaid-svg-iuNaeeLK2YoFKfao {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg…...
dedecms 织梦自定义表单留言增加ajax验证码功能
增加ajax功能模块,用户不点击提交按钮,只要输入框失去焦点,就会提前提示验证码是否正确。 一,模板上增加验证码 <input name"vdcode"id"vdcode" placeholder"请输入验证码" type"text&quo…...
抖音增长新引擎:品融电商,一站式全案代运营领跑者
抖音增长新引擎:品融电商,一站式全案代运营领跑者 在抖音这个日活超7亿的流量汪洋中,品牌如何破浪前行?自建团队成本高、效果难控;碎片化运营又难成合力——这正是许多企业面临的增长困局。品融电商以「抖音全案代运营…...
1.3 VSCode安装与环境配置
进入网址Visual Studio Code - Code Editing. Redefined下载.deb文件,然后打开终端,进入下载文件夹,键入命令 sudo dpkg -i code_1.100.3-1748872405_amd64.deb 在终端键入命令code即启动vscode 需要安装插件列表 1.Chinese简化 2.ros …...
第25节 Node.js 断言测试
Node.js的assert模块主要用于编写程序的单元测试时使用,通过断言可以提早发现和排查出错误。 稳定性: 5 - 锁定 这个模块可用于应用的单元测试,通过 require(assert) 可以使用这个模块。 assert.fail(actual, expected, message, operator) 使用参数…...
视频行为标注工具BehaviLabel(源码+使用介绍+Windows.Exe版本)
前言: 最近在做行为检测相关的模型,用的是时空图卷积网络(STGCN),但原有kinetic-400数据集数据质量较低,需要进行细粒度的标注,同时粗略搜了下已有开源工具基本都集中于图像分割这块,…...
JVM虚拟机:内存结构、垃圾回收、性能优化
1、JVM虚拟机的简介 Java 虚拟机(Java Virtual Machine 简称:JVM)是运行所有 Java 程序的抽象计算机,是 Java 语言的运行环境,实现了 Java 程序的跨平台特性。JVM 屏蔽了与具体操作系统平台相关的信息,使得 Java 程序只需生成在 JVM 上运行的目标代码(字节码),就可以…...
浪潮交换机配置track检测实现高速公路收费网络主备切换NQA
浪潮交换机track配置 项目背景高速网络拓扑网络情况分析通信线路收费网络路由 收费汇聚交换机相应配置收费汇聚track配置 项目背景 在实施省内一条高速公路时遇到的需求,本次涉及的主要是收费汇聚交换机的配置,浪潮网络设备在高速项目很少,通…...
jmeter聚合报告中参数详解
sample、average、min、max、90%line、95%line,99%line、Error错误率、吞吐量Thoughput、KB/sec每秒传输的数据量 sample(样本数) 表示测试中发送的请求数量,即测试执行了多少次请求。 单位,以个或者次数表示。 示例:…...
如何配置一个sql server使得其它用户可以通过excel odbc获取数据
要让其他用户通过 Excel 使用 ODBC 连接到 SQL Server 获取数据,你需要完成以下配置步骤: ✅ 一、在 SQL Server 端配置(服务器设置) 1. 启用 TCP/IP 协议 打开 “SQL Server 配置管理器”。导航到:SQL Server 网络配…...
