C++17 新特性解析:Lambda 捕获 this

C++17 引入了许多改进和新特性,其中之一是对 lambda 表达式的增强。在这篇文章中,我们将深入探讨 lambda 表达式中的一个特别有用的新特性:通过 *this 捕获当前对象的副本。这个特性不仅提高了代码的安全性,还极大地简化了某些场景下的编程模式。
- Lambda 表达式简介
Lambda 表达式是 C++11 中首次引入的一种匿名函数对象,它极大地简化了编程模式,特别是在使用 STL 算法或进行事件驱动编程时。Lambda 表达式的基本语法如下:
[捕获列表](参数列表) -> 返回类型 {函数体
};
- 捕获列表:用于捕获外部变量,使其在 lambda 表达式中可用。
- 参数列表:与普通函数类似,用于接收参数。
- 返回类型:可选,用于指定 lambda 的返回类型。
- 函数体:包含 lambda 的逻辑。
例如,以下是一个简单的 lambda 表达式,用于打印一个整数:
auto print = [](int x) {std::cout << x << std::endl;
};
print(42);
Lambda 表达式的强大之处在于它的灵活性和简洁性,它允许我们在需要的地方快速定义一个匿名函数,而无需单独声明一个函数对象。
- C++17 中的 *this 捕获
在 C++17 之前,如果你想在 lambda 表达式中使用当前类的成员变量或成员函数,你通常会捕获 this 指针。例如:
class MyClass {
public:int value = 10;void doSomething() {auto lambda = [this]() {std::cout << this->value << std::endl;};lambda();}
};
这种方式的问题是,它捕获的是 this 指针,而不是对象本身。这意味着如果外部对象的生命周期结束,而 lambda 表达式仍在使用,就可能访问到无效的内存。这种问题在多线程或异步编程中尤为常见,可能导致难以调试的错误。
为了解决这个问题,C++17 引入了通过 *this 捕获当前对象的副本的能力。这样,lambda 表达式就拥有了当前对象的一个完整副本,从而避免了潜在的悬挂指针问题。
示例代码
class MyClass {
public:int value = 10;void doSomething() {auto lambda = [*this]() {std::cout << value << std::endl; // 直接使用 value,不需要 this-> 前缀};lambda();}
};
在这个例子中,*this 在 lambda 表达式中创建了 MyClass 的一个副本,因此即使原始对象被销毁,lambda 表达式中的副本仍然是有效的。这种捕获方式不仅安全,还简化了代码的编写。
- 使用场景
*this 的捕获非常适合以下几种场景:
3.1 异步操作
在多线程或异步编程中,lambda 表达式可能会在不同的线程中执行。如果捕获的是 this 指针,而原始对象的生命周期结束,可能会导致未定义行为。通过捕获 *this,可以确保 lambda 表达式中使用的对象副本始终有效。
#include <iostream>
#include <thread>
#include <future>class MyClass {
public:int value = 10;void doSomething() {auto lambda = [*this]() {std::this_thread::sleep_for(std::chrono::seconds(1)); // 模拟异步操作std::cout << value << std::endl;};// 启动异步任务std::async(std::launch::async, lambda);}
};int main() {MyClass obj;obj.doSomething();return 0; // 对象 obj 的生命周期结束,但 lambda 中的副本仍然有效
}
3.2 避免悬挂指针
当你担心原始对象可能在 lambda 执行时被销毁,使用 *this 捕获可以安全地复制对象,避免访问已销毁的对象。
class MyClass {
public:int value = 10;void doSomething() {auto lambda = [*this]() {std::cout << value << std::endl;};lambda();}
};int main() {MyClass obj;auto lambda = obj.doSomething();// obj 的生命周期结束,但 lambda 中的副本仍然有效
}
3.3 值捕获的简化
直接通过 *this 捕获可以避免列出类中每个需要的成员。如果你的类中有多个成员变量或成员函数需要在 lambda 中使用,捕获 *this 是一种更简洁的方式。
class MyClass {
public:int value1 = 10;int value2 = 20;void doSomething() {auto lambda = [*this]() {std::cout << value1 + value2 << std::endl;};lambda();}
};
- 性能与注意事项
虽然 *this 捕获提供了极大的便利和安全性,但它也引入了一些性能开销。捕获 *this 会创建当前对象的一个副本,这意味着:
- 对象的拷贝构造函数会被调用。如果对象较大或拷贝构造函数较复杂,可能会导致性能下降。
- 内存占用增加。由于 lambda 中存储了对象的副本,因此需要更多的内存。
因此,在使用 *this 捕获时,需要权衡安全性和性能。如果对象较小且拷贝构造函数简单,*this 捕获是一个非常好的选择。但 if 对象较大或拷贝操作代价较高,可能需要考虑其他方式,例如手动管理对象的生命周期或使用智能指针。
- 总结
C++17 的 *this 捕获为 lambda 表达式提供了更大的灵活性和安全性。通过允许复制当前对象,它不仅简化了代码,还增强了程序的健壮性。这是 C++17 中众多改进中的一个亮点,值得每个 C++ 开发者了解和使用。
在实际开发中,合理利用 *this 捕获可以避免悬挂指针问题,简化异步编程的复杂性,并提高代码的可读性和安全性。当然,开发者也需要根据实际情况权衡性能和安全性,选择最适合的捕获方式。
希望这篇文章能帮助你更好地理解和使用 C++17 中的这一新特性。如果你有任何问题或建议,欢迎在评论区留言讨论!
相关文章:
C++17 新特性解析:Lambda 捕获 this
C17 引入了许多改进和新特性,其中之一是对 lambda 表达式的增强。在这篇文章中,我们将深入探讨 lambda 表达式中的一个特别有用的新特性:通过 *this 捕获当前对象的副本。这个特性不仅提高了代码的安全性,还极大地简化了某些场景下…...
Spring Boot 使用 Micrometer 集成 Prometheus 监控 Java 应用性能
在Spring Boot中使用Micrometer集成Prometheus来监控Java应用性能是一种常见的做法。 一、Micrometer简介 Micrometer是一个开源的Java项目,主要用于为JVM应用程序提供监控和度量功能。以下是对Micrometer的详细介绍: 定义与功能 Micrometer是一个针…...
Spring Boot 事件驱动:构建灵活可扩展的应用
在 Spring Boot 应用中,事件发布和监听机制是一种强大的工具,它允许不同的组件之间以松耦合的方式进行通信。这种机制不仅可以提高代码的可维护性和可扩展性,还能帮助我们构建更加灵活、响应式的应用。本文将深入探讨 Spring Boot 的事件发布…...
IM系统设计
读多写少,一般采用写扩散成timeline来做 写扩散模式 利用last message id作为这个作为最后一个消息体 timeline和批量未读和ack 利用ZSET来维护连接的定时心跳,来续约运营商的连接不断开...
华为EC6110T-海思Hi3798MV310_安卓9.0_通刷-强刷固件包
华为EC6110T-海思Hi3798MV310_安卓9.0_通刷-强刷固件包 刷机教程说明: 适用机型:华为EC6110-T、华为EC6110-U、华为EC6110-M 破解总分为两个部分:拆机短接破解(保留IPTV)和OTT卡刷(不保留IPTV)…...
ASP.NET Blazor托管模型有哪些?
今天我们来说说Blazor的三种部署方式,如果大家还不了解Blazor,那么我先简单介绍下Blazor Blazor 是一种 .NET 前端 Web 框架,在单个编程模型中同时支持服务器端呈现和客户端交互性: ● 使用 C# 创建丰富的交互式 UI。 ● 共享使用…...
PyTorch广告点击率预测(CTR)利用深度学习提升广告效果
目录 广告点击率预测问题数据集结构广告点击率预测模型的构建1. 数据集准备2. 构建数据加载器3. 构建深度学习模型4. 训练与评估 总结 广告点击率预测(CTR,Click-Through Rate Prediction)是在线广告领域中的重要任务,它帮助广告平…...
PAT甲级-1017 Queueing at Bank
题目 题目大意 银行有k个窗口,每个窗口只能服务1个人。如果3个窗口已满,就需要等待。给出n个人到达银行的时间和服务时间,要求计算每个人的平均等待时间。如果某个人的到达时间超过17:00:00,则不被服务,等待时间也不计…...
OneData体系架构详解
阿里巴巴的 OneData 体系架构方法论,主要分为三个阶段:业务板块、规范定义 和 模型设计。每个阶段的核心目标是确保数据的高效管理、共享与分析能力。 一. 业务板块(Business Segment) 业务板块是OneData体系架构中的第一步&…...
Gin 框架入门实战系列教程
一,Gin介绍 Gin是一个 Go (Golang) 编写的轻量级 http web 框架,运行速度非常快,如果你是性能和高效的追求者,我们推荐你使用Gin框架。 Gin最擅长的就是Api接口的高并发,如果项目的规模不大,业务相对简单…...
鸿蒙harmony json转对象(2)
在ArkTS(Ark TypeScript)中,接口(interface)是用来定义一个对象的结构,它可以包含属性、方法签名,以及嵌套的类型(包括其他接口或对象类型)。因此,接口里面可…...
M-LAG与E-trunk
M-LAG和E-trunk都是用来实现跨设备链路聚合,解决单点故障的,其大部分特性相同,工作模式M-LAG更胜一筹,支持双活,而且其原理感觉像是vrrpmstp的升级版,是往增加网络可靠性去发展的;而E-trunk是基于LACP扩展实现…...
【面试常见问题】
如何自我介绍 自我介绍是面试关键部分,是面试官了解求职者的首要途径,清晰自信的介绍能提升面试官印象,对求职成功至关重要。 糟糕的自我介绍示例 求职者朱晓明虽表明自己善于交际、积极,23 年毕业且从事 java 开发,…...
Spring Boot Starter介绍
前言 大概10来年以前,当时springboot刚刚出现并没有流行,当时的Java开发者们开发Web应用主要是使用spring整合springmvc或者struts、iBatis、hibernate等开发框架来进行开发。项目里一般有许多xml文件配置,其中配置了很多项目中需要用到的Be…...
vue和reacts数据响应式的差异
Vue 的数据响应式: 原理: Vue 使用 Object.defineProperty 或 Proxy(在 Vue 3 中)来实现数据的响应式。当创建 Vue 实例时,会对 data 对象中的属性进行遍历,将其转换为响应式属性。对于 Object.definePro…...
OpenEuler学习笔记(九):安装 OpenEuler后配置和优化
安装OpenEuler后,可以从系统基础设置、网络配置、性能优化等方面进行配置和优化,以下是具体内容: 系统基础设置 更新系统:以root用户登录系统后,在终端中执行sudo yum update命令,对系统进行更新…...
npm命令与yarn命令的区别
npm与Yarn的区别详解 在软件开发中,npm和Yarn都是流行的包管理工具,它们各自拥有独特的特性和优势。以下是它们的主要区别: 1. 安装速度 npm:安装速度相对较慢,尤其是在依赖项较多的情况下。Yarn:采用并…...
python如何导出数据到excel文件
python导出数据到excel文件的方法: 1、调用Workbook()对象中的add_sheet()方法 wb xlwt.Workbook() ws wb.add_sheet(A Test Sheet) 2、通过add_sheet()方法中的write()函数将数据写入到excel中,然后使用save()函数保存excel文件 ws.write(0, 0, 1234…...
MYSQL学习笔记(五):单行函数(字符串、数学、日期时间、条件判断、信息、加密、进制转换函数)讲解
前言: 学习和使用数据库可以说是程序员必须具备能力,这里将更新关于MYSQL的使用讲解,大概应该会更新30篇,涵盖入门、进阶、高级(一些原理分析);这一篇是讲解单行函数,当然mysql函数很多哈,只有多用才能记得…...
Grafana系列之Dashboard:新增仪表板、新增变量、过滤变量、变量查询、导入仪表板、变量联动、Grafana Alert
概述 关于Prometheus和Grafana的安装,略过。 写在前面 Dashboard:仪表板,可包含多个PanelPanel:面板,Dashboard中的组件 如有写得不对的地方,烦请指出。 新增仪表板 点击右上角的 选择New dashboard…...
基于距离变化能量开销动态调整的WSN低功耗拓扑控制开销算法matlab仿真
目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.算法仿真参数 5.算法理论概述 6.参考文献 7.完整程序 1.程序功能描述 通过动态调整节点通信的能量开销,平衡网络负载,延长WSN生命周期。具体通过建立基于距离的能量消耗模型&am…...
PHP和Node.js哪个更爽?
先说结论,rust完胜。 php:laravel,swoole,webman,最开始在苏宁的时候写了几年php,当时觉得php真的是世界上最好的语言,因为当初活在舒适圈里,不愿意跳出来,就好比当初活在…...
376. Wiggle Subsequence
376. Wiggle Subsequence 代码 class Solution { public:int wiggleMaxLength(vector<int>& nums) {int n nums.size();int res 1;int prediff 0;int curdiff 0;for(int i 0;i < n-1;i){curdiff nums[i1] - nums[i];if( (prediff > 0 && curdif…...
五年级数学知识边界总结思考-下册
目录 一、背景二、过程1.观察物体小学五年级下册“观察物体”知识点详解:由来、作用与意义**一、知识点核心内容****二、知识点的由来:从生活实践到数学抽象****三、知识的作用:解决实际问题的工具****四、学习的意义:培养核心素养…...
Mac软件卸载指南,简单易懂!
刚和Adobe分手,它却总在Library里给你写"回忆录"?卸载的Final Cut Pro像电子幽灵般阴魂不散?总是会有残留文件,别慌!这份Mac软件卸载指南,将用最硬核的方式教你"数字分手术"࿰…...
Spring AI 入门:Java 开发者的生成式 AI 实践之路
一、Spring AI 简介 在人工智能技术快速迭代的今天,Spring AI 作为 Spring 生态系统的新生力量,正在成为 Java 开发者拥抱生成式 AI 的最佳选择。该框架通过模块化设计实现了与主流 AI 服务(如 OpenAI、Anthropic)的无缝对接&…...
C++ Visual Studio 2017厂商给的源码没有.sln文件 易兆微芯片下载工具加开机动画下载。
1.先用Visual Studio 2017打开Yichip YC31xx loader.vcxproj,再用Visual Studio 2022打开。再保侟就有.sln文件了。 易兆微芯片下载工具加开机动画下载 ExtraDownloadFile1Info.\logo.bin|0|0|10D2000|0 MFC应用兼容CMD 在BOOL CYichipYC31xxloaderDlg::OnIni…...
以光量子为例,详解量子获取方式
光量子技术获取量子比特可在室温下进行。该方式有望通过与名为硅光子学(silicon photonics)的光波导(optical waveguide)芯片制造技术和光纤等光通信技术相结合来实现量子计算机。量子力学中,光既是波又是粒子。光子本…...
安宝特方案丨船舶智造的“AR+AI+作业标准化管理解决方案”(装配)
船舶制造装配管理现状:装配工作依赖人工经验,装配工人凭借长期实践积累的操作技巧完成零部件组装。企业通常制定了装配作业指导书,但在实际执行中,工人对指导书的理解和遵循程度参差不齐。 船舶装配过程中的挑战与需求 挑战 (1…...
免费PDF转图片工具
免费PDF转图片工具 一款简单易用的PDF转图片工具,可以将PDF文件快速转换为高质量PNG图片。无需安装复杂的软件,也不需要在线上传文件,保护您的隐私。 工具截图 主要特点 🚀 快速转换:本地转换,无需等待上…...
