当前位置: 首页 > news >正文

c++ lambda 表达式

1. 简介

lambda(匿名函数)是C++11引入的一种函数对象,它允许我们在需要函数的地方创建一个临时的、匿名的函数。lambda表达式表示一个可以执行的代码单元,可以理解为一个未命名的内联函数。Lambda函数可以用于简化代码、提高可读性,并且常与算法STL容器多线程等一起使用。

2. Lambda表达式的语法形式

[capture list] (parameter list) -> return type { function body }

capture list(捕获列表):可以捕获一定范围内的变量,可以为空。用于指定在lambda函数中使用的外部变量。
parameter list(参数列表):函数参数列表,可以为空。
return type(返回类型):指定lambda表达式返回值类型,可以省略,让编译器自动推导返回类型
function body(函数体):定义了lambda表达式执行的代码块。

2.1 简化

原函数:

int add(int a , int b){return a  + b;
}
void printword(){std::cout << "aa" << std::endl;
}

lambda 函数:

//1. 最完整的语法:
//[]()->返回值{};
//捕获列表 函数的参数->函数返回值{函数体};
[](int a , int b)->int{return a + b;} ;
//2. 简化 :返回值可以不用写,编译器可以推断返回值的类型
[](int a , int b){return a + b;};
//3. 简化,如果函数的参数没有,那么()也可以不用那些了。
[]{;};
例如:[]{std::cout << "hello world" << std::endl;};

3. 调用Lambda函数

[]{std::cout << "hello world" << std::endl;};

仅仅是定义了一个函数,这个函数不会被执行
要想调用这个函数,需要使用函数名字()来调用,或者使用函数指针来接收它然后调用。

3.1 函数名字() 调用Lambda函数

1. 无参数,无返回值
[]{std::cout << "hello world" << std::endl;}();
2. 有参数,有返回值
[](int a , int b){return a + b;}(10,20);

3.2 函数指针调用Lambda函数

1. 无参数,无返回值
void(*f)() = []{std::cout << "hello world" << std::endl;
};
f();
2. 有参数,有返回值
int(*fadd)(int,int)  =[](int a , int b){return a  + b;
};
cout <<  fadd(10,20) <<endl;

3.3 auto 调用Lambda函数

```cpp
1. 无参数,无返回值
auto f = []{std::cout << "hello world" << std::endl;
};
f();
2. 有参数,有返回值
auto fadd = [](int a , int b){return a  + b;
};
cout <<  fadd(10,20) <<endl;

4. 捕获列表的使用

lambda表达式需要在函数体中定义,这时如果想访问所处函数中的某个成员(局部变量),那么就需要使用捕获列表了。捕获列表的写法通常有以下几种形式:

形式作用
[a]表示值传递方式捕获变量 a
[=]表示值传递方式捕获所有父作用域的变量(包括this)
[&a]表示引用方式传递捕获变量a
[&]表示引用传递方式捕获所有父作用域的变量(包括this)
[this]表示值传递方式捕获当前的this指针
[=,&a,&b]引用方式捕获 a 和 b , 值传递方式捕获其他所有变量 (这是组合写法)

代码:

#include<iostream>using namespace std;int main(){int a = 20 ;int b = 30;//值传递方式捕获 a 和 b,在lambda 函数体内无法修改a和b的值auto f1 = [a,b]{return a + b;}; cout << f1()  << endl; //引用方式捕获 a 和 b  , 可以在内部修改a 和 b的值auto f2 = [&a,&b]{ a = 40; //这里修改会导致外部的a 也跟着修改。return a + b;};cout << f2()  << endl; //这里打印35cout << "a= "<< a << endl; //再打印一次,a 变成30了return 0 ;
}

4. 特点

匿名性:Lambda函数是匿名的,没有函数名。它的定义和使用都是在需要的地方,无需提前声明。
闭包(Closure):Lambda函数可以捕获外部作用域的变量,包括局部变量、函数参数、全局变量等,并在函数体内使用。捕获方式可通过捕获列表进行指定。捕获的变量可以在函数体内进行读取和修改。
自动推导:Lambda函数的参数类型和返回类型可以通过编译器自动推导,可以省略。如果省略返回类型,则根据函数体中的返回语句自动推导返回类型。
函数对象:Lambda函数实际上是一个函数对象,可以像普通函数一样被调用。可以将Lambda函数作为参数传递给其他函数,也可以存储在容器中。
代码:

#include <iostream>
#include <vector>
#include <algorithm>int main() {std::vector<int> numbers = {1, 2, 3, 4, 5};// Lambda函数作为参数传递给std::for_each算法std::for_each(numbers.begin(), numbers.end(), [](int num) {std::cout << num << " ";});// Lambda函数捕获外部变量,并进行计算int factor = 2;std::transform(numbers.begin(), numbers.end(), numbers.begin(), [factor](int num) {return num * factor;});// Lambda函数作为谓词进行排序std::sort(numbers.begin(), numbers.end(), [](int a, int b) {return a > b;});// 输出结果for (int num : numbers) {std::cout << num << " ";}return 0;
}

std::for_each 算法中,使用Lambda函数作为参数,遍历 numbers 容器并输出每个元素。
std::transform 算法中,使用Lambda函数对 numbers 容器中的每个元素进行乘以 factor 的操作。
std::sort 算法中,使用Lambda函数作为谓词,根据元素的大小进行降序排序。

5. 使用场景

  1. 标准算法:作为标准算法的参数,用于定制算法的行为。例如,使用Lambda函数作为谓词对容器进行筛选、排序等操作
#include <iostream>
#include <vector>
#include <algorithm>int main() {std::vector<int> numbers = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9};// 使用Lambda函数作为谓词,筛选出偶数auto isEven = [](int num) {return num % 2 == 0;};std::vector<int> evenNumbers;std::copy_if(numbers.begin(), numbers.end(), std::back_inserter(evenNumbers), isEven);// 输出结果for (int num : evenNumbers) {std::cout << num << " ";}return 0;
}
  1. 回调函数:作为回调函数传递给其他函数,用于处理特定事件或条件。
#include <iostream>
#include <functional>// 模拟事件处理器
void processEvent(int data, std::function<void(int)> callback) {// 处理事件// ...// 调用回调函数callback(data);
}int main() {int eventData = 0;// 使用Lambda函数作为回调函数processEvent(eventData, [](int data) {std::cout << "Event processed with data: " << data << std::endl;});return 0;
}

或者:

#include <iostream>
#include <functional>void performOperation(int a, int b, std::function<void(int)> callback) {int result = a + b;callback(result);
}int main() {int x = 5, y = 3;performOperation(x, y, [](int result) {std::cout << "The result is: " << result << std::endl;});return 0;
}
  1. 多线程编程:作为线程的执行体,方便地捕获外部变量。
#include <iostream>
#include <thread>int main() {int value = 42;// 使用Lambda函数作为线程的执行体std::thread t([&value]() {// 在子线程中使用外部变量std::cout << "Value in thread: " << value << std::endl;});// 等待子线程结束t.join();return 0;
}
  1. 函数对象的临时定义:在需要临时定义函数对象的场景,避免为简单的操作额外定义函数。
#include <iostream>
#include <vector>
#include <algorithm>int main() {std::vector<int> numbers = {3, 1, 4, 1, 5};// 使用Lambda函数作为函数对象std::for_each(numbers.begin(), numbers.end(), [](int num) {std::cout << num << " ";});return 0;
}

相关文章:

c++ lambda 表达式

1. 简介 lambda&#xff08;匿名函数&#xff09;是C11引入的一种函数对象&#xff0c;它允许我们在需要函数的地方创建一个临时的、匿名的函数。lambda表达式表示一个可以执行的代码单元&#xff0c;可以理解为一个未命名的内联函数。Lambda函数可以用于简化代码、提高可读性…...

Go语言入门心法(七): 并发与通道

Go语言入门心法(一): 基础语法 Go语言入门心法(二): 结构体 Go语言入门心法(三): 接口 Go语言入门心法(四): 异常体系 Go语言入门心法(五): 函数 一: go语言并发与通道...

前端组件封装:构建模块化、可维护和可重用的前端应用

前端组件封装&#xff1a;构建模块化、可维护和可重用的前端应用 前端开发领域的快速演进已经将前端应用的规模和复杂性提升到了一个新的水平。在这个背景下&#xff0c;前端组件封装成为了一项关键实践&#xff0c;旨在构建模块化、可维护和可重用的前端应用。在本文中&#…...

GPT绘制流程图咒语

【咒语】下面是我的一篇论文选取部分&#xff0c;为了让读者更好理解&#xff0c;我准备画一张图&#xff0c;请你阅读后为我设计一下这个图应该怎么画&#xff0c;更有说服力&#xff0c;更容易理解 论文片段&#xff1a; 多模态数据融合研究的基础在于有效的数据采集。首先&a…...

【扩散模型从原理到实战】Chapter1 扩散模型简介

文章目录 1.1 扩散模型的原理生成模型扩散过程DDPM的扩散过程前向过程反向过程优化目标 1.2 扩散模型的发展开始扩散&#xff1a;DDPM加速生成&#xff1a;采样器刷新记录&#xff1a;基于CLIP的多模态图像生成引爆网络&#xff1a;基于CLIP的多模态图像生成再次“出圈”&#…...

使用轮廓分数提升时间序列聚类的表现

我们将使用轮廓分数和一些距离指标来执行时间序列聚类实验&#xff0c;并且进行可视化 让我们看看下面的时间序列: 如果沿着y轴移动序列添加随机噪声&#xff0c;并随机化这些序列&#xff0c;那么它们几乎无法分辨&#xff0c;如下图所示-现在很难将时间序列列分组为簇: 上面…...

蔬菜水果生鲜配送团购商城小程序的作用是什么

蔬菜水果是人们生活所需品&#xff0c;从业者众多&#xff0c;无论小摊贩还是超市商场都有不少人每天光临&#xff0c;当然这些只是自然流量&#xff0c;在实际经营中&#xff0c;蔬菜水果商家还是面临着一些难题。 对蔬菜水果商家而言&#xff0c;线下门店是重要的&#xff0…...

金融用户实践|分布式存储支持数据仓库业务系统性能验证

作者&#xff1a;深耕行业的 SmartX 金融团队 闫海涛 估值是指对资产或负债的价值进行评估的过程&#xff0c;这对于投资决策具有重要意义。每个金融公司资管业务人员都期望能够实现实时的业务估值&#xff0c;快速获取最新的数据和指标&#xff0c;从而做出更明智的投资决策。…...

代码随想录二刷 Day41

509. 斐波那契数 这个题简单入门&#xff0c;注意下N小于等于1的情况就可以 class Solution { public:int fib(int n) {if (n < 1) return n; //这句不写的话test能过但是另外的过不了vector<int> result(n 1); //定义存放dp结果的数组&#xff0c;还要定义大小r…...

C++项目实战——基于多设计模式下的同步异步日志系统-⑪-日志器管理类与全局建造者类设计(单例模式)

文章目录 专栏导读日志器建造者类完善单例日志器管理类设计思想单例日志器管理类设计全局建造者类设计日志器类、建造者类整理日志器管理类测试 专栏导读 &#x1f338;作者简介&#xff1a;花想云 &#xff0c;在读本科生一枚&#xff0c;C/C领域新星创作者&#xff0c;新星计…...

Hadoop3教程(十四):MapReduce中的排序

文章目录 &#xff08;99&#xff09;WritableComparable排序什么是排序什么时候需要排序排序有哪些分类如何实现自定义排序 &#xff08;100&#xff09;全排序案例案例需求思路分析实际代码 &#xff08;101&#xff09;二次排序案例&#xff08;102&#xff09; 区内排序案例…...

测试需要写测试用例吗?

如何理解软件的质量 我们都知道&#xff0c;一个软件从无到有要经过需求设计、编码实现、测试验证、部署发布这四个主要环节。 需求来源于用户反馈、市场调研或者商业判断。意指在市场行为中&#xff0c;部分人群存在某些诉求或痛点&#xff0c;只要想办法满足这些人群的诉求…...

Qt 视口和窗口的区别

视口和窗口 绘图设备的物理坐标是基本的坐标系&#xff0c;通过QPainter的平移、旋转等变换可以得到更容易操作的逻辑坐标 为了实现更方便的坐标&#xff0c;QPainter还提供了视口(Viewport)和窗口(Window)坐标系&#xff0c;通过QPainter内部的坐标变换矩阵自动转换为绘图设…...

使用Git将GitHub仓库下载到本地

前记&#xff1a; git svn sourcetree gitee github gitlab gitblit gitbucket gitolite gogs 版本控制 | 仓库管理 ---- 系列工程笔记. Platform&#xff1a;Windows 10 Git version&#xff1a;git version 2.32.0.windows.1 Function&#xff1a;使用Git将GitHub仓库下载…...

前端需要了解的浏览器缓存知识

文章目录 前言为什么需要缓存&#xff1f;DNS缓存缓存读写顺序缓存位置memory cache&#xff08;浏览器本地缓存&#xff09;disk cache&#xff08;硬盘缓存&#xff09;重点&#xff01;&#xff01;&#xff01; 缓存策略 - 强缓存和协商缓存1&#xff09;强缓存ExpiresCach…...

自动驾驶:控制算法概述

自动驾驶&#xff1a;控制算法概述 常见控制算法PID算法LQR算法MPC算法 自动驾驶控制算法横向控制纵向控制 参考文献 常见控制算法 PID算法 PID&#xff08;Proportional-Integral-Derivative&#xff09;控制是一种经典的反馈控制算法&#xff0c;通常用于稳定性和响应速度要…...

【Mysql】Mysql的字符集和比较规则(三)

字符集和比较规则简介 字符集简介 我们知道在计算机中只能以二进制的方式对数据进行存储&#xff0c;那么他们之间是怎样对应并进行转换的&#xff1f;我们需要了解两个概念&#xff1a; 字符范围&#xff1a;我们可以将哪些字符转换成二进制数据&#xff0c;也就是规定好字…...

【SpringCloud-11】SCA-sentinel

sentinel是一个流量控制、熔断降级的组件&#xff0c;可以替换第一代中的hystrix。 hystrix用起来没有那么方便&#xff1a; 1、要在调用方引入hystrix&#xff0c;没有ui界面进行配置&#xff0c;需要在代码中进行配置&#xff0c;侵入了业务代码。 2、还要自己搭建监控平台…...

设计模式:简单工厂模式(C#、JAVA、JavaScript、C++、Python、Go、PHP):

简介&#xff1a; 简单工厂模式&#xff0c;它提供了一个用于创建对象的接口&#xff0c;但具体创建的对象类型可以在运行时决定。这种模式通常用于创建具有共同接口的对象&#xff0c;并且可以根据客户端代码中的参数或配置来选择要创建的具体对象类型。 在简单工厂模式中&am…...

浅谈智能照明控制系统在智慧建筑中的应用

贾丽丽 安科瑞电气股份有限公司 上海嘉定 201801 摘要&#xff1a;新时期&#xff0c;建筑行业发展迅速&#xff0c;在信息化背景下&#xff0c;建筑功能逐渐拓展&#xff0c;呈现了智能化的发展态势。智能建筑更加安全、节能、环保&#xff0c;也符合绿色建筑理念。在建筑智…...

AI-调查研究-01-正念冥想有用吗?对健康的影响及科学指南

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; &#x1f680; AI篇持续更新中&#xff01;&#xff08;长期更新&#xff09; 目前2025年06月05日更新到&#xff1a; AI炼丹日志-28 - Aud…...

【Python】 -- 趣味代码 - 小恐龙游戏

文章目录 文章目录 00 小恐龙游戏程序设计框架代码结构和功能游戏流程总结01 小恐龙游戏程序设计02 百度网盘地址00 小恐龙游戏程序设计框架 这段代码是一个基于 Pygame 的简易跑酷游戏的完整实现,玩家控制一个角色(龙)躲避障碍物(仙人掌和乌鸦)。以下是代码的详细介绍:…...

Mybatis逆向工程,动态创建实体类、条件扩展类、Mapper接口、Mapper.xml映射文件

今天呢&#xff0c;博主的学习进度也是步入了Java Mybatis 框架&#xff0c;目前正在逐步杨帆旗航。 那么接下来就给大家出一期有关 Mybatis 逆向工程的教学&#xff0c;希望能对大家有所帮助&#xff0c;也特别欢迎大家指点不足之处&#xff0c;小生很乐意接受正确的建议&…...

蓝牙 BLE 扫描面试题大全(2):进阶面试题与实战演练

前文覆盖了 BLE 扫描的基础概念与经典问题蓝牙 BLE 扫描面试题大全(1)&#xff1a;从基础到实战的深度解析-CSDN博客&#xff0c;但实际面试中&#xff0c;企业更关注候选人对复杂场景的应对能力&#xff08;如多设备并发扫描、低功耗与高发现率的平衡&#xff09;和前沿技术的…...

MODBUS TCP转CANopen 技术赋能高效协同作业

在现代工业自动化领域&#xff0c;MODBUS TCP和CANopen两种通讯协议因其稳定性和高效性被广泛应用于各种设备和系统中。而随着科技的不断进步&#xff0c;这两种通讯协议也正在被逐步融合&#xff0c;形成了一种新型的通讯方式——开疆智能MODBUS TCP转CANopen网关KJ-TCPC-CANP…...

leetcodeSQL解题:3564. 季节性销售分析

leetcodeSQL解题&#xff1a;3564. 季节性销售分析 题目&#xff1a; 表&#xff1a;sales ---------------------- | Column Name | Type | ---------------------- | sale_id | int | | product_id | int | | sale_date | date | | quantity | int | | price | decimal | -…...

AI书签管理工具开发全记录(十九):嵌入资源处理

1.前言 &#x1f4dd; 在上一篇文章中&#xff0c;我们完成了书签的导入导出功能。本篇文章我们研究如何处理嵌入资源&#xff0c;方便后续将资源打包到一个可执行文件中。 2.embed介绍 &#x1f3af; Go 1.16 引入了革命性的 embed 包&#xff0c;彻底改变了静态资源管理的…...

CVPR2025重磅突破:AnomalyAny框架实现单样本生成逼真异常数据,破解视觉检测瓶颈!

本文介绍了一种名为AnomalyAny的创新框架&#xff0c;该方法利用Stable Diffusion的强大生成能力&#xff0c;仅需单个正常样本和文本描述&#xff0c;即可生成逼真且多样化的异常样本&#xff0c;有效解决了视觉异常检测中异常样本稀缺的难题&#xff0c;为工业质检、医疗影像…...

nnUNet V2修改网络——暴力替换网络为UNet++

更换前,要用nnUNet V2跑通所用数据集,证明nnUNet V2、数据集、运行环境等没有问题 阅读nnU-Net V2 的 U-Net结构,初步了解要修改的网络,知己知彼,修改起来才能游刃有余。 U-Net存在两个局限,一是网络的最佳深度因应用场景而异,这取决于任务的难度和可用于训练的标注数…...

精益数据分析(98/126):电商转化率优化与网站性能的底层逻辑

精益数据分析&#xff08;98/126&#xff09;&#xff1a;电商转化率优化与网站性能的底层逻辑 在电子商务领域&#xff0c;转化率与网站性能是决定商业成败的核心指标。今天&#xff0c;我们将深入解析不同类型电商平台的转化率基准&#xff0c;探讨页面加载速度对用户行为的…...