【C++】lambda表达式、generalized lambda-capture

  C++11中新加入了lambda表达式,很是方便,最常见的使用就是在 STL 的 sort 函数中直接写排序方法,如下:

#include 
#include 
#include 
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 
#include 
#include 
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,广义捕获。

你可能感兴趣的:(#,C++)