我的编程空间,编程开发者的网络收藏夹
学习永远不晚

C++匿名函数lambda详解

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

C++匿名函数lambda详解

匿名函数lambda

一、匿名函数的基本语法

[捕获列表](参数列表) mutable(可选) 异常属性 -> 返回类型 {// 函数体}

语法规则:lambda表达式可以看成是一般函数的函数名被略去,返回值使用了一个 -> 的形式表示。唯一与普通函数不同的是增加了“捕获列表”。

// lambda_test lambda_test.cc#include using namespace std;void test01(){cout << "test01" << endl;auto Add = [](int a, int b) -> int {return a + b;};cout << Add(1, 2) << endl;}int main(int argc,char **argv){test01();return 0;}

编译(要指定-std=c++11):

g++ -o lambda_test lambda_test.cc -std=c++11

输出结果:

$ ./lambda_testtest013

一般情况下,编译器可以自动推断出lambda表达式的返回类型,所以我们可以不指定返回类型,即:

// lambda_test lambda_test.cc#include using namespace std;void test02(){cout << "test02" << endl;auto Add = [](int a, int b){return a + b;};cout << Add(1, 2) << endl;}int main(int argc,char **argv){//test01();test02();return 0;}

但是如果函数体内有多个return语句时,编译器无法自动推断出返回类型,此时必须指定返回类型。

二、捕获列表

有时候,需要在匿名函数内使用外部变量,所以用捕获列表来传递参数。根据传递参数的行为,捕获列表可分为以下几种:

2.1、值捕获

与参数传值类似,值捕获的前提是变量可以拷贝,不同之处则在于,被捕获的变量在 lambda表达式被创建时拷贝,而不是在调用时才拷贝:

// lambda_test lambda_test.cc#include using namespace std;void test03(){cout << "test03" << endl;int c = 20;int d = 30;auto Add = [c,d](int a, int b) {cout << "d = "<< d << endl;return c;};d = 10; // 在这里修改 d 的值,会改变 Add里的 d 值吗?cout << Add(1, 2) << endl;}int main(int argc,char **argv){//test01();//test02();test03();return 0;}

执行结果:

$ ./lambda_testtest03d = 3020

2.2、引用捕获

与引用传参类似,引用捕获保存的是引用,值会发生变化。

#include using namespace std;void test04(){cout << "test04" << endl;int c = 20;int d = 30;auto Add = [c, &d](int a, int b) {cout << "c = " << c << endl;cout << "d = " << d << endl;return c;};d = 10;//在这里修改d的值,会改变Add里的d值吗?cout << Add(1, 2) << endl;}int main(int argc,char **argv){//test01();//test02();//test03();test04();return 0;}

执行结果:

$ ./lambda_testtest04c = 20d = 1020

2.3、隐式捕获

手动书写捕获列表有时候是非常复杂的,这种机械性的工作可以交给编译器来处理,这时候可以在捕获列表中写一个 & 或 = 向编译器声明采用引用捕获或者值捕获。编译器会将外部变量全部捕获。

#include using namespace std;void test05(){cout << "test05" << endl;int c = 20;int d = 30;auto Add = [&](int a, int b) {cout << "c = " << c << endl;cout << "d = " << d << endl;return c;};d = 10;//在这里修改d的值,会改变Add里的d值吗?cout << Add(1, 2) << endl;}void test06(){cout << "test06" << endl;int c = 20;int d = 30;auto Add = [=](int a, int b) {cout << "c = " << c << endl;cout << "d = " << d << endl;return c;};d = 10;//在这里修改d的值,会改变Add里的d值吗?cout << Add(1, 2) << endl;}int main(int argc,char **argv){//test01();//test02();//test03();test05();test06();return 0;}

输出:

$ ./lambda_testtest05c = 20d = 1020test06c = 20d = 3020

2.4、空捕获列表

捕获列表’[]'中为空,表示Lambda不能使用所在函数中的变量。

void test07(){cout << "test07" << endl;int c = 20;int d = 30;auto Add = [](int a, int b) {cout << "c = " << c << endl; // 编译报错cout << "d = " << d << endl; // 编译报错return c; // 编译报错};d = 10;cout << Add(1, 2) << endl;}

编译报错:

lambda_test.cc:95:14: note: the lambda has no capture-default  auto Add = [](int a, int b) {              ^lambda_test.cc:93:6: note: ‘int c’ declared here  int c = 20;      ^lambda_test.cc:97:21: error: ‘d’ is not captured   cout << "d = " << d << endl;

2.5、表达式捕获

上面提到的值捕获、引用捕获都是已经在外层作用域声明的变量,因此这些捕获方式捕获的均为左值,而不能捕获右值。

C++14之后支持捕获右值,允许捕获的成员用任意的表达式进行初始化,被声明的捕获变量类型会根据表达式进行判断,判断方式与使用 auto 本质上是相同的:

#include #include using namespace std;void test08(){cout << "test08" << endl;auto important = make_unique<int>(1);auto Add = [v1 = 1, v2 = std::move(important)](int a, int b)->int{return a + b + v1 + (*v2);};cout << Add(1, 2) << endl;}int main(int argc,char **argv){test08();return 0;}

执行结果:

$ ./lambda_testtest085

2.6、泛型 Lambda

在C++14之前,lambda表示的形参只能指定具体的类型,没法泛型化。从 C++14 开始, Lambda 函数的形式参数可以使用 auto关键字来产生意义上的泛型。
简单点说,就是通过auto使lambda自适应参数类型:

#include using namespace std;void test09(){cout << "test09" << endl;auto Add = [](auto a, auto b) {return a + b;};cout << Add(1, 2) << endl;cout << Add(1.1, 2.2) << endl;}int main(int argc,char **argv){test09();return 0;}

执行结果:

./lambda_testtest0933.3

2.7、可变lambda

(1)采用值捕获的方式,lambda不能修改其值,如果想要修改,使用mutable修饰。
(2)采用引用捕获的方式,lambda可以直接修改其值。

#include using namespace std;void test10(){cout << "test10" << endl;int v = 10;// 值捕获方式,使用mutable修饰,可以改变捕获的变量值auto tes = [v]() mutable {return ++v;};v = 5;auto a = tes();// a=11;cout << a << endl;}void test11(){cout << "test11" << endl;int v = 10;auto Add = [&v]{return v++;};v = 6;cout << Add() << endl;}int main(int argc,char **argv){test10();test11();return 0;}

执行结果:

$ ./lambda_testtest1011test116

2.8、混合捕获

  1. 要求捕获列表中第一个元素必须是隐式捕获(&或=)。
  2. 混合使用时,若隐式捕获采用引用捕获(&)则显式捕获的变量必须采用值捕获的方式。
  3. 若隐式捕获采用值捕获(=),则显式捕获的变量必须采用引用捕获的方式。
#include using namespace std;void test12(){cout << "test12" << endl;int c = 12;int d = 30;int e = 30;// auto Add = [&, d, e](int a, int b)auto Add = [=, &c](int a, int b) -> int {c = a;cout << "d=" << d << ", e=" << e << endl;return c;};d = 20;cout << Add(1, 2) << endl;cout << "c:" << c << endl;}int main(int argc,char **argv){test12();return 0;}

测试结果:

$ ./lambda_testtest12d=30, e=301c:1

2.10、Lambda捕获列表总结

捕获含义
[]空捕获列表,Lambda不能使用所在函数中的变量。
[names]names是一个逗号分隔的名字列表,这些名字都是Lambda所在函数的局部变量。默认情况下,这些变量会被拷贝,然后按值传递,名字前面如果使用了&,则按引用传递
[&]隐式捕获列表,Lambda体内使用的局部变量都按引用方式传递
[=]隐式捕获列表,Lanbda体内使用的局部变量都按值传递
[&,identifier_list]identifier_list是一个逗号分隔的列表,包含0个或多个来自所在函数的变量,这些变量采用值捕获的方式,其他变量则被隐式捕获,采用引用方式传递,identifier_list中的名字前面不能使用&。
[=,identifier_list]identifier_list中的变量采用引用方式捕获,而被隐式捕获的变量都采用按值传递的方式捕获。identifier_list中的名字不能包含this,且这些名字面前必须使用&。

总结

  1. lambda表达式的目的是把函数写的更加内聚;只需要在内部使用,就没必要写到外部,干扰其他函数,同时使代码更简洁。
  2. 如果捕获列表为[&],则表示所有的外部变量都按引用传递给lambda使用。
  3. 如果捕获列表为[=],则表示所有的外部变量都按值传递给lambda使用。
  4. 匿名函数构建的时候对于按值传递的捕获列表,会立即将当前可以取到的值拷贝一份作为常数,然后将该常数作为参数传递。

后言

本专栏知识点是通过<零声教育>的系统学习,进行梳理总结写下文章,对c/c++linux系统提升感兴趣的读者,可以点击链接,详细查看详细的服务:C/C++服务器课程

来源地址:https://blog.csdn.net/Long_xu/article/details/127869979

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

C++匿名函数lambda详解

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

C++11中匿名函数lambda的使用详解

我最早接触lambda的概念是在matlab中,那时候在做数值模拟的课题,lambda可以快速定义简单的函数,当时觉得好方便。任何语言都有这个功能,下面来看看C++11新引入的lambda是如何使用的吧
2023-05-18

C#中的匿名函数、lambda表达式解读

这篇文章主要介绍了C#中的匿名函数、lambda表达式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
2023-01-28

Python基础:lambda 匿名函数

格式lambda argument1, argument2,... argumentN : expressionsquare = lambda x: x**2print(square(2))与常规函数区别  匿名函数 lambda 和常规函
2023-01-31

C++lambda函数详解

小编可以明确告诉大家:lambda函数是C++11中最重要的,使用最广泛的,最具现代风格的内容,lambda函数的出现改变了C++编程的思维方式。所以快和小编学习一下C++11中lambda函数的使用吧
2023-02-13

C++11中的lambda匿名函数怎么使用

本篇内容主要讲解“C++11中的lambda匿名函数怎么使用”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“C++11中的lambda匿名函数怎么使用”吧!lambda 源自希腊字母表中第 11
2023-07-04

 Python匿名函数lambda怎么用

今天小编给大家分享一下 Python匿名函数lambda怎么用的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。1.前言在 Py
2023-06-29

Python匿名函数lambda的用法

本篇内容主要讲解“Python匿名函数lambda的用法”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Python匿名函数lambda的用法”吧!Python内置了多个关键字,今天我们来介绍下其
2023-06-02

python如何使用匿名函数lambda

这篇文章主要介绍python如何使用匿名函数lambda,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!匿名函数 lambda所谓匿名,意即不再使用 def 语句这样标准的形式定义一个函数。lambda 只是一个表达式
2023-06-03

Python 3 之 lambda匿名函

------- lambda -------------------------------------    除了def语句之外,Python还提供了一种生成函数对象的表达式形式。由于它与LISP语言中的一个工具很相似,所以称为lambd
2023-01-31

python基础教程之匿名函数lambda

python lambda 当我们在使用函数时,有时候,并不需要显示的定义一个函数,我们可以使用匿名函数更加方便,在Python中对匿名函数也提供了支持。比如当我们想计算两个数a,b之和时,即f(a,b) = a + b。我们可以有两种方法
2022-06-04

scala匿名函数案例详解

Scala支持一级函数,函数可以用函数文字语法表达,即(x:Int)=>x+1,该函数可以由一个叫作函数值的对象来表示,这篇文章主要介绍了scala匿名函数详解,需要的朋友可以参考下
2023-03-19

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录