C++匿名函数lambda详解

这篇具有很好参考价值的文章主要介绍了C++匿名函数lambda详解。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一、匿名函数的基本语法

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

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

// lambda_test lambda_test.cc
#include <iostream>

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_test
test01
3

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

// lambda_test lambda_test.cc
#include <iostream>

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 <iostream>

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_test
test03
d = 30
20

2.2、引用捕获

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

#include <iostream>

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_test
test04
c = 20
d = 10
20

2.3、隐式捕获

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

#include <iostream>

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_test
test05
c = 20
d = 10
20
test06
c = 20
d = 30
20

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 <iostream>
#include <memory>
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_test
test08
5

2.6、泛型 Lambda

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

#include <iostream>

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_test
test09
3
3.3

2.7、可变lambda

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

#include <iostream>

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_test
test10
11
test11
6

2.8、混合捕获

  1. 要求捕获列表中第一个元素必须是隐式捕获(&或=)。
  2. 混合使用时,若隐式捕获采用引用捕获(&)则显式捕获的变量必须采用值捕获的方式。
  3. 若隐式捕获采用值捕获(=),则显式捕获的变量必须采用引用捕获的方式。
#include <iostream>

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_test
test12
d=30, e=30
1
c: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++匿名函数lambda详解文章来源地址https://www.toymoban.com/news/detail-439273.html

到了这里,关于C++匿名函数lambda详解的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • python 匿名函数(lambda函数)

    Python中的匿名函数是指没有命名标识符的函数,通常被称为lambda函数。与普通函数不同,它们是一种更加简洁的方式来编写小型临时函数。在Python中,匿名函数使用 lambda 来定义,其语法如下: 其中, arguments 表示函数参数,可以是一个或多个,多个参数之间用逗号分隔

    2024年02月02日
    浏览(50)
  • python匿名函数Lambda

    Lambda函数也被称为匿名(没有名称)函数,它直接接受参数的数量以及使用该参数执行的条件或操作,该参数以冒号分隔,并返回最终结果。为了在大型代码库上编写代码时执行一项小任务,或者在函数中执行一项小任务,便在正常过程中使用lambda函数。 argument_list是参数列表,

    2024年02月16日
    浏览(48)
  • lambda匿名函数

    问题: 什么是lambda函数?它有什么好处?举例说明 解答 含义 在Python中,不通过def来声明函数名字,而是通过 lambda 来定义的函数称为匿名函数,即函数没有具体的名称,你可以理解为 一句话写一个函数 Lambda表达式是Python中一类特殊的定义函数的形式,从语义上讲,它们

    2024年02月15日
    浏览(59)
  • 【C#进阶】C#中的委托、事件、回调函数、匿名函数和lambda表达式

    委托是一种类型,它可以存储对一个或多个方法的引用。它类似于C/C++中的函数指针,允许您将方法作为参数传递、存储和调用。 写法: delegate return_type delegate_name( ); return_type :表示委托所引用方法的返回类型。 delegate_name :表示委托的名称。 parameters :表示委托所引用方法

    2024年02月06日
    浏览(56)
  • python基础----05-----函数的多返回值、函数的多种参数使用形式、函数作为参数传递、lambda匿名函数

    分为以下四种。 位置参数调用函数时根据函数定义的参数位置来传递参数,传递的参数和定义的参数的顺序及个数必须一致。 函数调用时通过“键=值”形式传递参数。 作用:可以让函数更加清晰、容易使用,同时也清楚了参数的顺序需求。 注意: 函数调用时,如果有位置参

    2024年02月08日
    浏览(51)
  • 面试之快速学习c++11- 列表初始化和 lambda匿名函数的定义

    学习地址: http://c.biancheng.net/view/3730.html 我们知道,在 C++98/03 中的对象初始化方法有很多种,请看下面的代码: 2 .为了统一初始化方式,并且让初始化行为具有确定的效果,C++11 中提出了列表初始化(List-initialization)的概念。 3 . 在上面我们已经看到了,对于普通数组和

    2024年02月13日
    浏览(40)
  • 【C++】C++11语法 ~ lambda 表达式

    (꒪ꇴ꒪(꒪ꇴ꒪ )🐣,我是 Scort 目前状态:大三非科班啃C++中 🌍博客主页:张小姐的猫~江湖背景 快上车🚘,握好方向盘跟我有一起打天下嘞! 送给自己的一句鸡汤🤔: 🔥真正的大师永远怀着一颗学徒的心 作者水平很有限,如果发现错误,可在评论区指正,感谢🙏 🎉🎉

    2024年01月20日
    浏览(60)
  • 【C++】C++ 引用详解 ① ( 变量的本质 - 引入 “ 引用 “ 概念 | 引用语法简介 | 引用做函数参数 | 复杂类型引用做函数参数 )

    \\\" 引用 \\\" 语法 是 C++ 语言中 特有的 , 在 C 语言中是没有 引用 这个概念的 ; 分析 引用 之前 , 先回顾下 变量 : 在 【C 语言】变量本质 ( 变量概念 | 变量本质 - 内存空间别名 | 变量存储位置 - 代码区 | 变量三要素 ) 博客中 , 介绍了变量的本质 : 变量 的本质是 内存空间 的 \\\" 别名

    2024年02月11日
    浏览(48)
  • C++类和对象-多态->多态的基本语法、多态的原理剖析、纯虚函数和抽象类、虚析构和纯虚析构

    #includeiostream using namespace std; //多态 //动物类 class Animal { public:     //Speak函数就是虚函数     //函数前面加上virtual,变成虚函数,那么编译器在编译的时候就不能确定函数调用了。     virtual void speak()     {         cout \\\"动物在说话\\\" endl;     } }; //猫类 class Cat

    2024年02月20日
    浏览(37)
  • 【C++】lambda表达式语法详细解读(代码演示,要点解析)

    前言 大家好吖,欢迎来到 YY 滴C++系列 ,热烈欢迎! 本章主要内容面向接触过C++的老铁 主要内容含: 欢迎订阅 YY 滴C++专栏!更多干货持续更新!以下是传送门! YY的《C++》专栏 YY的《C++11》专栏 YY的《Linux》专栏 YY的《数据结构》专栏 YY的《C语言基础》专栏 YY的《初学者易

    2024年02月03日
    浏览(48)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包