c++ Lambda表达式

本文介绍了Lambda表达式的基本概念、语法形式、应用示例,包括捕获列表、参数列表、表达式修饰符以及返回值类型。重点阐述了Lambda如何工作,以及它在STL算法中的应用。

Lambda表达式简介

        Lambda有很多叫法,有Lambda表达式、Lambda函数、匿名函数。Lambda表达式是现代C++在C ++ 11和更高版本中的一个新的语法糖。

        lambda表达式(也称为lambda函数)是在调用或作为函数参数传递的位置处定义匿名函数对象的便捷方法。通常,lambda用于封装传递给算法或异步方法的几行代码。

Lambda表达式语法形式

[外部变量捕获列表] (表达式参数列表) 表达式修饰符 ->返回值类型 {表达式体};

Lambda表达式应用示例

//计算容器vec中大于3的元素个数
int test(void)
{
	vector<int> vec={1,2,3,4,5,6};
	int CountNum = count_if(vec.begin(),vec.end(),[](int val){return val>3;});   //传入的是lambda表达式,返回的是bool类型,其中val为遍历vec容器传入的元素值
}
PS:可以给Lambda表达式起个名字

虽然lambda表达式是匿名函数,但是实际上也可以给lambda表达式指定一个名称,如下表示:

auto f = [](int x ){return x % 3 ==0;};

此后再需要使用该lambda表达式,就可以使用f()来代替。举一个例子:

#include <iostream>
using namespace std;
int main()
{
    int a = 5,b = 6;
    auto f = [=]{return a+b;};//[=]按值捕获了a和b
    cout << f() << endl;
    return 0;
}

Lambda表达式的优缺点

优点:

  • 可以直接在需要调用函数的位置定义短小精悍的函数,而不需要预先定义好函数
std::find_if(v.begin(), v.end(), [](int& item){return item > 2});
  • 使用Lamdba表达式变得更加紧凑,结构层次更加明显、代码可读性更好

缺点:

  • Lamdba表达式语法比较灵活,增加了阅读代码的难度
  • 对于函数复用无能为力

        

Lambda表达式语法形式详解

[外部变量捕获列表] (表达式参数列表) 表达式修饰符 ->返回值类型 {表达式体};

外部参数捕获列表

        捕获列表能够捕捉上文中的局部变量以供Lambda函数在函数体中使用,当然,只能使用到定义该Lambda表达式为止定义过的局部变量。

基本语法:

[ ]:代表不捕获Lambda表达式外的变量;
&:代表以引用传递的方式捕获Lambda表达式外的变量;
=:代表以值传递的方式捕获Lambda表达式外的局部变量,即以const引用的方式传值

this:表示Lambda表达式可以使用Lambda表达式所在类的成员变量;
 

基本语法应用:

[a]或[=a]:表示以值传递的方式传递变量a ,即const int a,在函数体内不可改变a的值;但是可以对Lambda表达式使用mutable修饰符修饰,使得函数对象参数可以进行赋值,但是该函数对象参数不是被修改为引用传递方式。
[&a]:表示以引用传递的方式传递变量a,在函数体内可以改变a的值;
[x,&y]:x为值传递方式,y为引用传值方式;
[=,&x,&y]:除x,y为引用传递方式以外,其他参数都为值传递方式进行传递;
[&,x,y]:除x,y为值传递方式以外,其他参数都为引用传递方式进行传递;

注意点:

不过值得注意的是,捕捉列表不允许变量重复捕获。下面一些例子就是典型的重复,会导致编译时期的错误。例如:

  • [=,a]这里已经以值传递方式捕捉了所有变量,但是重复捕捉a了,会报错的;
  • [&,&this]这里&已经以引用传递方式捕捉了所有变量,再捕捉this也是一种重复。

下面针对 捕获列表 的使用举一些实际例子:

[=]举例:

 [=]表示值传递方式捕获所有父作用域的变量(包括this

int index = 1;
int num = 100;
auto function = ([=]{
			std::cout << "index: "<< index << ", " 
                << "num: "<< num << std::endl;
	}
);

function();

[a]举例:

int num = 100;
auto function = ([num]{
		std::cout << num << std::endl;
	}
);

function();

 [&]举例:

 [&]表示引用传递方式捕捉所有父作用域的变量(包括this

int index = 1;
int num = 100;
auto function = ([&]{
		num = 1000;
		index = 2;
		std::cout << "index: "<< index << ", " 
            << "num: "<< num << std::endl;
	}
);

function();

 [this]举例:

 [this]表示值传递方式捕捉当前的this指针

#include <iostream>
using namespace std;
 
class Lambda
{
public:
    void sayHello() {
        std::cout << "Hello" << std::endl;
    };

    void lambda() {
        auto function = [this]{ 
            this->sayHello(); 
        };

        function();
    }
};
 
int main()
{
    Lambda demo;
    demo.lambda();
}

参数列表

        与普通函数的参数列表基本一样,如果不需要参数传递,则可以连同括号“()”一起省略。

auto function = [] (int first, int second){
    return first + second;
};
	
function(100, 200);
auto f = [] {
	cout << "Hello" << '\n';
};

表达式修饰符

这一部分是可以省略的,常见的修饰符有如下两个:

mutable

        默认情况下Lambda函数总是一个const函数,mutable修饰符可以取消其常量性。在使用该修饰符时,参数列表不可省略(即使参数为空)。

#include <iostream>
using namespace std;

int main()
{
   int m = 0;
   int n = 0;
   [&, n] (int a) mutable { m = ++n + a; }(4);//n可正常被修改
   cout << m << endl << n << endl;
}
异常说明 throw() 

        你可以使用 throw() 异常规范来指示 Lambda 表达式不会引发任何异常。与普通函数一样,如果 Lambda 表达式声明 C4297 异常规范且 Lambda 体引发异常,Visual C++ 编译器将生成警告 throw() 。

int main() // C4297 expected 
{ 
 	[]() throw() { throw 5; }(); 
}

在MSDN的异常规范中,明确指出异常规范是在 C++11 中弃用的 C++ 语言功能。因此这里不建议大家使用。

 

返回值类型

        Lambda表达式的返回类型会自动推导,除非你指定了返回类型,否则不必使用关键字,这部分常常省略(连同符号”->”一起省略)。

auto x1 = [](int i){ return i; };

表达式体

        内容与普通函数一样,区别是除了可以使用参数之外,还可以使用所有捕获的变量。

总结一下,表达式体作用域内一共可以访问:

  • 捕获变量
  • 形参变量
  • 局部声明的变量
  • 类数据成员,当在类内声明this并被捕获时
  • 具有静态存储持续时间的任何变量,例如全局变量
#include <iostream>
using namespace std;

int main()
{
   int m = 0;
   int n = 0;
   [&, n] (int a) mutable { m = ++n + a; }(4);
   cout << m << endl << n << endl;
}

Lambda表达式工作原理

        编译器会把一个Lambda表达式生成一个仿函数。

        回顾,大家在使用 c++ sort算法的时候,参3会要求传递一个谓词。这个谓词可以是个仿函数,也可以是个Lambda表达式。也可以看出,二者本质是很接近的。所以,Lamdba表达式可以在STL算法库中经常使用。

示例:

auto print = []{cout << "Hello World!" << endl; };

编译器会把上面这一句翻译为下面的代码:

class print_class
{
public:
	void operator()(void) const
	{
		cout << "Hello World!" << endl;
	}
};
// 用构造的类创建对象,print此时就是一个函数对象
auto print = print_class();

不了解 仿函数 的小伙伴可顺便参考:c++仿函数及在STL中的应用-优快云博客


参考

c++ lambda 看这篇就够了!(有点详细)_c++ 运行时 构建 lamda-优快云博客

C++Lambda表达式,超详细的讲解,保证一遍懂_c++ lambda-优快云博客

C++ Lambda表达式详解-优快云博客

### C++ Lambda 表达式用法详解 Lambda 表达式C++11 引入的重要特性之一,它允许开发者在调用或传递函数对象时直接定义匿名函数,极大提升了代码的简洁性和可读性。Lambda 表达式广泛应用于 STL 算法、异步编程、事件回调等场景中,是现代 C++ 编程的核心工具之一。 #### 基本语法结构 一个完整的 Lambda 表达式通常由以下几个部分组成: - **捕获列表**(Capture List):指定 Lambda 表达式如何访问外部变量。 - **参数列表**(Parameter List):可选,用于指定 Lambda 接受的参数。 - **可变说明符**(mutable):允许修改按值捕获的变量副本。 - **异常说明符**(Exception Specifier):可选,指定 Lambda 是否抛出异常。 - **返回类型**(Return Type):可选,若未指定,编译器会自动推导。 - **函数体**(Function Body):包含具体的实现逻辑。 示例代码如下: ```cpp auto lambda = [](int a) -> int { return a + 1; }; std::cout << lambda(1) << std::endl; // 输出: 2 ``` #### 捕获机制 Lambda 表达式的捕获机制决定了它如何访问和修改外部变量,主要有以下几种方式: - **按值捕获**([=]):捕获外部变量的副本,Lambda 内部无法直接修改外部变量。若需修改副本,需使用 `mutable` 关键字。 - **按引用捕获**([&]):捕获外部变量的引用,Lambda 内部可直接修改外部变量。 - **显式捕获**:可精确指定捕获的变量及方式,例如 `[x, &y]` 表示按值捕获 `x`,按引用捕获 `y`。 示例代码如下: ```cpp int x = 10; auto lambda = [=]() mutable { x++; std::cout << x << std::endl; // 输出 11 }; lambda(); std::cout << x << std::endl; // 输出 10(外部 x 未变) ``` #### 生命周期管理 使用 Lambda 表达式时,尤其需要注意按引用捕获带来的生命周期问题。如果 Lambda 被存储并在捕获变量的生命周期结束后调用,可能导致未定义行为。因此,开发者需确保引用的变量在 Lambda 调用时仍然有效。 #### 性能考量 按值捕获大对象可能会带来拷贝开销,建议使用按引用捕获或智能指针来避免不必要的性能损耗。此外,隐式捕获(如 [=] 和 [&])仅捕获 Lambda 体内实际使用的外部变量,因此可以有效减少捕获范围。 #### 应用场景 Lambda 表达式在 STL 算法中表现尤为突出,例如 `std::for_each`、`std::transform` 等算法中可以使用 Lambda 来简化函数对象的定义。此外,Lambda 还广泛应用于异步编程、事件回调、闭包管理等场景中。 示例代码如下: ```cpp #include <algorithm> #include <vector> #include <iostream> int main() { std::vector<int> vec = {1, 2, 3, 4, 5}; std::for_each(vec.begin(), vec.end(), [](int x) { std::cout << x << " "; }); std::cout << std::endl; return 0; } ``` #### 标准演进与未来趋势 随着 C++ 标准的不断演进,Lambda 表达式的功能也在不断增强。例如,C++20 引入了模式匹配(Pattern Matching),进一步扩展了 Lambda 的应用场景。未来,Lambda 表达式将继续成为现代 C++ 开发的核心工具之一。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值