C++11中新加入了lambda表达式,很是方便,最常见的使用就是在 STL 的 sort 函数中直接写排序方法,如下:
#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
int main()
{
vector<int> v{ 1, 5, 3, 2, 4 };
sort(v.begin(), v.end(), [](const int& a, const int& b) { return a < b; });
for (const auto& vi : v) cout << vi << endl; // output: 1 2 3 4 5
return 0;
}
最近在使用的过程中发现了一个问题,比如我想利用 vector 的第一个元素对后边所有元素进行排序的话,需要在 [] 中capture 第一个元素的值,因为函数参数并没有传这个值。C++ lambda 表达式中 [] 可以按值捕获,也可以引用捕获(当然也可以都有,一起用):
1、[=a] 就是按值捕获外部变量 a,lambda 表达式中对 a 进行修改不影响外边的 a,相当于正常函数的 passed by value;
2、[=] 就是在这个 lambda 表达式同一作用域的所有参数都按值捕获;
3、[&a] 就是按引用捕获外部变量 a,lambda 表达式中对 a 进行修改会影响外边的 a,相当于正常函数的 passed by reference
4、[&] 就是在这个 lambda 表达式同一作用域的所有参数都引用捕获
但是对数组来说,想传入第一个元素,不能 [v[0]],因为 v[0] 是表达式,不能传入表达式,需要传值,但是又需要传第一个元素的值,可以先用一个临时变量获取,然后传入这个临时变量,或者直接在 [] 里边传入一个临时变量。
#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
int main()
{
vector<int> v{ -1, 5, 3, 2, 4 };
// Wrong: [v0 = v[0]] or [*v.begin()] // 不能传表达式
// Right: [v0 = -1] or [v0 = v[0]] or [v0 = *v.begin()]
sort(v.begin() + 1, v.end(), [v0 = v[0]](const int& a, const int& b) { return a * v0 < b * v0; });
for (const auto& vi : v) cout << vi << endl; // output: -1 5 4 3 2
return 0;
}
这个方法应该是在C++14里边新加入的特性,叫做 generalized lambda-capture,广义捕获。