Lambda表达式在c++++中是一种简洁定义匿名函数对象的方式,常用于需要简单函数逻辑的地方。它的语法形式为capture -> return_type { function_body },其中capture指定捕获外部变量的方式,parameters是参数列表,return_type可省略由编译器推导,function_body为函数体。常见用法包括:1. 配合stl算法如std::sort使用;2. 作为回调函数;3. 替代轻量级functor。捕获列表可根据需求选择[=]按值捕获所有变量,[&]按引用捕获,或指定捕获特定变量如[var]、[&var]、[this]。返回类型通常可自动推导,但若存在多个不一致的return语句则需显式声明以避免歧义。例如lambda表达式在排序时可替代比较函数,捕获变量时需根据是否希望反映外部变化选择值或引用捕获,且注意生命周期问题,确保代码安全与效率。
Lambda表达式在c++中是一种简洁定义匿名函数对象的方式,常用于需要简单函数逻辑的地方,比如作为参数传给算法函数(如std::sort、std::for_each)或者事件回调等。
它的语法形式大致如下:
[capture](parameters) -> return_type { function_body }
其中:
立即学习“C++免费学习笔记(深入)”;
- capture:捕获外部变量的方式,可以是值捕获(=)、引用捕获(&)或指定捕获某些变量。
- parameters:参数列表,和普通函数的参数写法一样。
- return_type:返回类型可省略,编译器会自动推导。
- function_body:函数体内容。
基本用法:替代简单的函数对象
比如排序一个数组,你想按降序排,但不想专门写个比较函数。这时候可以用lambda:
std::vector<int> nums = {3, 1, 4, 2}; std::sort(nums.begin(), nums.end(), [](int a, int b) { return a > b; });
这里 lambda 表达式 [](int a, int b) { return a > b; } 就充当了比较函数的作用。
常见使用场景包括:
- 配合STL算法使用,如find_if、transform等。
- 作为回调函数传递给其他模块。
- 在需要轻量级函数对象时替代functor。
捕获列表怎么选?看你要不要用外部变量
lambda 的一大特点是能“捕获”它所在作用域中的变量。捕获方式有几种,常见的有:
- [=]:以值的方式捕获所有外部变量。
- [&]:以引用的方式捕获所有外部变量。
- [var]:只捕获特定变量,按值。
- [&var]:只捕获特定变量,按引用。
- [this]:捕获当前类的this指针,用于访问成员变量。
举个例子:
int x = 10; auto f = [x](int y) { return x + y; }; std::cout << f(5); // 输出15
这个lambda捕获了x的值,之后即使x变了,f里的x还是原来的10。
如果你希望它能修改x,就得用引用捕获:
int x = 10; auto f = [&x](int y) { return x + y; }; x = 20; std::cout << f(5); // 输出25
所以什么时候用值捕获,什么时候用引用,要看你是否希望lambda内部反映外部变量的变化。
返回类型可以省略,但有时得显式写出来
大多数情况下,lambda的返回类型可以由编译器自动推导,不需要写-> return_type这部分。
但如果函数体内有多个return语句,且返回类型不一致,就需要明确写出返回类型,否则编译器可能无法推导。
例如:
auto func = [](bool flag) -> int { if (flag) return 1; else return 0.5; // 这里会转换成int };
虽然返回的实际上是double,但因为返回类型指定为int,结果会被转换。
这种时候明确返回类型是为了避免歧义或隐式转换带来的问题。
基本上就这些。lambda表达式在C++11引入后非常常用,掌握好捕获方式和基本结构,就能写出更简洁清晰的代码。不过也要注意别滥用,比如捕获this时要小心生命周期问题,否则容易出现悬空引用。