​C++ 中的 Lambda​

news2024/9/16 21:16:58

C++11引入了lambda表达式,使得C++程序员能够编写更简洁的回调函数和闭包,

Lambda 表达式用于定义并创建匿名的函数对象,以简化编程工作。

1  语法

[捕获列表] (参数) mutable 或 exception 声明 -> 返回值类型 {函数体}
//计算两个值的和
auto func = [](int a, int b) -> int{return a+b;};
//当返回值的类型是确定时,可以忽略返回值
auto func = [](int a, int b){return a + b;};
//调用
int sum = func(1, 3);

2 语法分析

2.1 捕获列表

Lambda 表达式相当于一个类,那么捕获列表就是传递给这个类的类成员。比如:

class Labmda
{
public:
    const int test;
    Labmda(int value):test(value)
    {
    }
public:
    int run(int a, int b)
    {
        return a + b + test;
    }
}

int main()
{
    int test = 10;
    auto func = Labmda(test);
    int sum = func.run(1, 3);
}

//使用Lambda 表达式的写法
int main()
{
    int test = 10;
    auto func = [test](int a, int b){return a + b + test;};
    int sum = func(1, 3);
}

捕获列表有以下格式:

版本一:
  • []:默认不捕获任何变量;
  • [=]:Lambda表达式之前的局部变量,包括所在类的this,变量按值方式传递,默认以值捕获所有变量;
  • [&]:Lambda表达式之前的局部变量,包括所在类的this,变量按引用方式传递,默认以引用捕获所有变量;
  • [x]:Lambda表达式之前的局部变量x的值,也可以传入多个值,如[a , b],仅以值捕获x,其它变量不捕获;
  • [&x]:Lambda表达式之前的局部变量x的引用,仅以引用捕获x,其它变量不捕获;
  • [=, &x]:默认以值捕获所有变量,但是x是例外,通过引用捕获;
  • [&, x]:默认以引用捕获所有变量,但是x是例外,通过值捕获;
  • [this]:Lambda表达式所在类的this,通过引用捕获当前对象(其实是复制指针);
  • [*this]:通过传值方式捕获当前对象;
版本二:

空:代表不捕获Lambda表达式外的变量;
&:代表以引用传递的方式捕获Lambda表达式外的变量;
=:代表以值传递的方式捕获Lambda表达式外的变量,即以const引用的方式传值;
this:表示Lambda表达式可以使用Lambda表达式所在类的成员变量;
a或=a:表示以值引用的方式传递变量a aa,即const int a,在函数体内不可改变a的值;但是可以对Lambda表达式使用mutable修饰符修饰,使得函数对象参数可以进行赋值,但是该函数对象参数不是被修改为引用传递方式,下面进行细说;
&a:表示以引用传递的方式传递变量a aa,在函数体内可以改变a的值;
x,&y:x为值传递方式,y为引用传值方式;
=,&x,&y:除x,y为引用传递方式以外,其他参数都为值传递方式进行传递;
&,x,y:除x,y为值传递方式以外,其他参数都为引用传递方式进行传递;
 

2.2 关键字声明

关键字声明一般都很少用到,也不建议随便使用,可以忽略不计。

mutable

当捕获列表以值的方式传递时有效,加上此关键字后,可以修改Lambda类成员(带const修饰符)。比如:

int test = 10;
//编译报错,test成员不能修改
auto func = [test](int a, int b){test = 8; return a + b + test;}; 
//编译正常
auto func = [test](int a, int b)mutable {test = 8; return a + b + test;}; 

这里需要注意的是:Lambda类成员test修改之后,并不会改变外部int test的值。

exception

exception 声明用于指定函数抛出的异常,如抛出整数类型的异常,可以使用 throw(int)

2.3 示例

捕获列表按值传递
int test = 10;
auto func = [=](int a, int b){return a + b + test;};
auto func2 = [test](int a, int b){return a + b + test;};
int sum = func(1, 3); //sum等于14

这里需要注意的是func表达式中test的值只更新到表达式之前:

int test = 10;
auto func = [=](int a, int b){return a + b + test;};
test = 5;
int sum = func(1, 3); //sum还是等于14
捕获列表按引用传递
int test = 10;
auto func = [&](int a, int b){test = 5; return a + b + test;};
auto func2 = [&test](int a, int b){test = 5; return a + b + test;};
int sum = func(1, 3); //sum等于9,test等于5

这里func表达式中test的值就能随时更新:

int test = 10;
auto func = [&](int a, int b){return a + b + test;};
test = 5;
int sum = func(1, 3); //sum等于9,test等于5

using camera_output_target_ptr =
    std::unique_ptr<ACameraOutputTarget, void (*)(ACameraOutputTarget*)>; auto target= camera_output_target_ptr{
        [=]() {
            ACameraOutputTarget* target{};
            ACameraOutputTarget_create(
                    window, addressof(target));
            return target;
        }(),
        ACameraOutputTarget_free
    };      camera_output_target_ptr lambda表达式后面为什么要加()

等价于

ANativeWindow* window, 指针传递

auto create_target = [=]() {  
    ACameraOutputTarget* target = nullptr;  
    ACameraOutputTarget_create(window, &target); // 确保使用正确的地址传递  
    return target;  
};  
  
auto target = camera_output_target_ptr(create_target(), ACameraOutputTarget_free);

3 注意事项

lambda表达式提供了一种定义匿名函数对象的方式,这些对象可以捕获其所在作用域中的变量。Lambda表达式的参数捕获机制允许你选择是按值捕获还是按引用捕获这些变量。这里有一些在使用时需要注意的地方:

3.1 参数捕获方式

  • 按值捕获 ([=]):lambda表达式内部将使用捕获变量的副本。如果捕获的变量在lambda表达式外部被修改,lambda表达式内部使用的是修改前的值。
  • 按引用捕获 ([&]):lambda表达式内部将直接引用捕获的变量。如果捕获的变量在lambda表达式外部被修改,lambda表达式内部将看到这些修改。
  • 明确捕获:你也可以明确指定哪些变量按值捕获,哪些按引用捕获(例如,[&, x] 表示按引用捕获所有变量,但x按值捕获;[=, &y] 表示按值捕获所有变量,但y按引用捕获)。

3.2 注意事项

  • 生命周期问题:如果lambda表达式按引用捕获了一个变量,并且这个lambda表达式在原始变量的生命周期之后被调用,那么这将导致未定义行为(通常是访问悬挂指针或引用)。
  • 修改捕获的变量:如果lambda表达式需要修改捕获的变量(按值捕获的除外),则这些变量必须按引用捕获。但是,这也要求调用者注意lambda表达式的副作用。
  • 性能考虑:按值捕获可能导致不必要的复制或移动操作,特别是当捕获大型对象时。按引用捕获可能更有效率,但需要注意生命周期问题。
  • 闭包:Lambda表达式与其捕获的变量一起形成了一个闭包。闭包中的变量在lambda表达式的作用域之外是不可见的,但lambda表达式可以访问和修改这些变量(取决于捕获方式)。
  • 默认捕获:如果不指定捕获列表(例如,[]),则不会捕获任何外部变量。这可能会导致编译错误,如果lambda表达式内部试图访问外部变量的话。
  • mutable关键字:默认情况下,lambda表达式的捕获列表中的变量在lambda体内是只读的(如果它们是按值捕获的)。如果你需要在lambda体内修改这些变量(即使是按值捕获的),你需要在lambda表达式的参数列表后添加mutable关键字。然而,这通常与按引用捕获一起使用更有意义。

int main() {  
    int x = 10;  
    auto lambda = [&]() { x = 20; }; // 按引用捕获,可以修改x  
    lambda();  
    std::cout << x << std::endl; // 输出 20  
  
    int y = 10;  
    auto lambda2 = [y]() mutable { y = 20; }; // 错误,y是按值捕获的,不能修改  
    // lambda2(); // 如果取消注释,将编译失败  
  
    // 正确的做法(如果需要修改)  
    auto lambda3 = [&y]() { y = 20; }; // 按引用捕获y  
    lambda3();  
    // 注意:这里的y是main函数中的y的引用,但如果lambda3的作用域超出了main函数,则会导致问题  
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2083349.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Great Wall长城工作站安装银河麒麟V10(SP1)-ARM版桌面操作系统

长城工作站安装银河麒麟V10(SP1)桌面操作系统 1. 硬件信息 [1]. Great Wall 长城台式微型计算机 产品型号&#xff1a;世恒TD120A2 型号代码&#xff1a;世恒TD120A2-019 电源&#xff1a;220V~3A 50Hz [2]. 芯片型号 架构&#xff1a; aarch64 CPU 运行模式&#xff1a…

【Qt窗口】—— 浮动窗口

目录 1.1 浮动窗口的创建 1.2 设置停靠的位置 1.3 示例小结 在Qt中&#xff0c;浮动窗口也称之为铆接部件&#xff0c;俗称为子窗口&#xff0c;浮动窗口是通过QDockWidget类来实现浮动的功能。浮动窗口⼀般是位于核⼼部件的周围&#xff0c;可以有多个。 1.1 浮动窗口的…

LeetCode 热题100-39 对称二叉树

对称二叉树 给你一个二叉树的根节点 root &#xff0c; 检查它是否轴对称。 示例 1&#xff1a; 输入&#xff1a;root [1,2,2,3,4,4,3] 输出&#xff1a;true示例 2&#xff1a; 输入&#xff1a;root [1,2,2,null,3,null,3] 输出&#xff1a;false提示&#xff1a; 树中…

【python报错已解决】AttributeError: module ‘PIL.Image‘ has no attribute ‘ANTIALIAS‘

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 引言 当我们使用某些Python库&#xff0c;如Pillow&#xff08;PIL的一个分支&#xff09;&#xff0c;进行图像处理时&#x…

设备共享租赁小程序系统开发制作方案

设备共享租赁小程序系统让用户方便地租赁或出租各类设备&#xff0c;包括但不限于工具、电子产品、运动器材等&#xff0c;以满足临时使用需求&#xff0c;同时为设备所有者创造额外收益。 目标用户 个人用户&#xff1a;需要临时使用工具、车辆等设备的个人。 企业用户&#…

STM32(F103ZET6)第十九课:FreeRtos的移植和使用

目录 需求一、FreeRtos简介二、移植FreeRtos1.复制代码2.内存空间分配和内核相关接口3.FreeRtosConfig4.添加到工程中三、任务块操作1.任务四种状态2.创建任务过程 需求 1.将FreeRtos&#xff08;嵌入式实时操作系统&#xff09;移植到STM32中。 2.在该系统中实现任务的创建、…

git学习教程--分支操作+远程仓库相关过程详述

目录 1.分支 1.1查看已有分支 1.2新的分支的创建 1.3改变指针的指向 1.4合并分支 2.删除分支 3.合并冲突 3.1一个简单操作 3.2手动解决冲突 4.git分支管理策略 4.1fast-forward模式 4.2no-ff模式 4.3总结 5.bug修复建议 6.强制删除 7.分布式版本控制系统 7.1远…

【递归回溯之floodfill算法专题练习】

1. 图像渲染 class Solution {int dx[4] {0, 0, -1, 1};int dy[4] {1, -1, 0, 0};int m, n;int oldcolor; public:vector<vector<int>> floodFill(vector<vector<int>>& image, int sr, int sc, int color) {oldcolor image[sr][sc]; // 保存原…

Java常用API(BigDecimal)

用于小数的精确计算 用来表示很大的小数 构造方法获取BigDecimal对象 public BigDecimal(double val) public BigDecimal(string val) 静态方法获取BigDecimal对象 public static BigDecimal value0f(double val) 1.通过传递double类型的小数来创建对象 这种方式有可能…

Linux|软件开发的基础概念|软件的源码本地编译和交叉编译概念

前言&#xff1a; 本文主要讲述软件的源码本地编译和交叉编译的基本概念&#xff0c;首先&#xff0c;是介绍什么是本地编译&#xff0c;什么是交叉编译&#xff0c;其次&#xff0c;本地编译和交叉编译到底是有什么用处&#xff0c;最后是交叉编译和本地编译的具体应用场景 …

边听边打?不再是难题,4款音频转文字神器推荐

无论是会议记录、课堂笔记还是采访录音&#xff0c;能快速准确地转录成文本&#xff0c;那可是大大提高了工作效率。市面上有几款工具在这方面做得不错&#xff0c;比如365在线转文字、布谷鸟配音、腾讯云语音识别和Speechnotes。今天就来个大比拼&#xff0c;看看它们各自的表…

人机交互的频率、时长、周期

人机交互的频率是指用户与系统互动的频繁程度&#xff1b;时长是每次互动的持续时间&#xff1b;周期是指在特定时间段内进行互动的规律或间隔。人机交互的频率、时长和周期通常与以下因素有关&#xff1a; &#xff08;1&#xff09;任务复杂性&#xff1a;复杂任务需要更多的…

docker部署clickhouse

1. 创建相关配置目录 mkdir -P /data/clickhouse/data mkdir -P /data/clickhouse/conf mkdir -P /data/clickhouse/log 2. 拉取镜像 # 下载最新版本clickhouse docker pull clickhouse/clickhouse-server # 下载指定版本clickhouse docker pull clickhouse/clickhouse…

电商数据分析:如何抓住关键指标提高销售额

在电商运营中&#xff0c;数据分析是不可或缺的一环。通过精准的数据分析&#xff0c;商家可以更好地了解市场动态、优化运营策略&#xff0c;从而提升销售业绩。然而&#xff0c;很多运营者在面对海量数据时常常无从下手。那么&#xff0c;电商运营到底该如何进行数据分析&…

Chapter 05 计算属性

欢迎大家订阅【Vue2Vue3】入门到实践 专栏&#xff0c;开启你的 Vue 学习之旅&#xff01; 文章目录 前言一、基础语法二、计算属性vs方法三、完整写法 前言 Vue.js 提供了丰富的功能&#xff0c;帮助开发者高效地构建用户界面。本篇文章详细讲解了其计算属性的基本语法、应用…

笔记整理—uboot启动过程(7)malloc初始化与内存环境变量

上一章说到了env环境变量并对前两章有关init_sequence部分做了总结&#xff0c;这一章将要对uboot部分的malloc初始化以及内存环境变量进行相关的说明。 mem_malloc_init是用于初始化uboot堆管理器的。自己维护了一段内存&#xff0c;就可用进行malloc和free的操作了。那么这个…

Mac/Linux系统matplotlib中文支持问题

背景 matplotlib是python中最常用的数据可视化分析工具&#xff0c;Mac和Linux系统无中文字体&#xff0c;不支持中文显示&#xff08;希望后续可以改进&#xff09;&#xff0c;需要进行字体的下载和设置才能解决。笔者经过实践&#xff0c;发现Mac系统和Linux系统解决方案略…

数据结构算法基础-单链表的新建(头插法、尾插法)

1.头插法 2.尾插法 3.代码及运行结果 设输入的值为&#xff1a;3 4 5 6 7&#xff08;到9999终止读值&#xff09; #include <stdio.h> #include <stdlib.h> typedef int ElemType;typedef struct LNode{ElemType data;struct LNode *next; }LNode,*LinkList;LinkL…

02 vue3之ref全局桶

ref 接受一个内部值并返回一个响应式且可变的 ref 对象。ref 对象仅有一个 .value property&#xff0c;指向该内部值。 <template><div class"">Ref:{{ name.a }}</div><button click"change()">change</button> </te…

如何用Java构建学生档案管理系统:实现学生信息的高效管理

✍✍计算机编程指导师 ⭐⭐个人介绍&#xff1a;自己非常喜欢研究技术问题&#xff01;专业做Java、Python、微信小程序、安卓、大数据、爬虫、Golang、大屏等实战项目。 ⛽⛽实战项目&#xff1a;有源码或者技术上的问题欢迎在评论区一起讨论交流&#xff01; ⚡⚡ Java实战 |…