一、认识lambda函数
[capture](parameters) mutable->return-type{statement}
上面是lambda函数的完整表达式。lambda函数,又叫lambda表达式,是一种匿名函数。即没有函数名的函数。所以读者完全是可以将lambda函数看成是一般函数来认识和使用。
不过,我们在使用的时候不会如上述那般复杂的定义一个lambda函数 [&a](int b, int c)->int{return a = b +c; }。而是使用如下较为省略的定义:
[&a](int b, int c){ return a = b + c; }
lambda函数和一般函数是一样的,只不过C++使其省略一些不必要的步骤:
int a = 10;
int func(int b, int c)
{
return a = b + c;
}
- [capture]:捕捉列表,可以捕捉(其实就是使用的意思)该lambda函数作用域内的变量。
- (parameters):参数列表,就是一般函数的参数列表。
- {statement}:函数体,就是一般函数的函数体。
可以看到,lambda函数省略了函数名,省略了返回类型改由编译器自动推导,改写了对函数外的变量的使用方式用[]进行捕捉.
捕捉方式有以下几种: - [&]:表示以引用的方式捕获作用域内的所有变量,包括this。
- [=]:表示以值的方式捕获作用域内的所有变量,包括this。
- [&a]:表示以引用的方式捕获作用域内的变量a。
- [a]:表示以值的方式捕获作用域内的变量a。
- [&, a]:表示以值的方式捕获变量a,以引用的方式捕获作用域内的所有变量。
- [=, &a]:表示以引用的方式捕获变量a,以值的方式捕获作用域内的所有变量。
二、lambda函数的使用和使用场景
1.当函数体内仅有一句或者两句语句时,建议使用lambda函数,可以减少代码量。
/*不使用lambda函数时,遍历对象元素*/
void func(int i)
{
std::cout << i << " ";
}
int main()
{
std::vector<int> ivec{1, 2, 3, 4, 5};
std::for_each(ivec.begin(), ivec.end(), func);
}
/*使用lambda函数时*/
int main()
{
std::vector<int> ivec{1, 2, 3, 4, 5};
std::for_each(ivec.begin(), ivec.end(), [](int i){ std::cout << i << " "; });
}
2.当使用std::bind( ) 或者 boost::bind( ) 时,建议使用lambda函数,可以使调用回调函数语句更加清晰明白显而易见。
boost::asio::write(socket, boost::asio::buffer(sbuf),
std::bind(&tcp::handle_write, this, auto error, auto bytes_transferred));
/*使用lambda函数*/
boost::asio::write(socket, boost::asio::buffer(sbuf),
[this](auto error, auto bytes){ this->handle_wtite(error, bytes); });
三、lambda函数的注意事项
- lambda函数捕捉变量时,捕捉值和捕捉引用的区别就是传参传值和传引用的区别。捕捉值不会改变原变量。
- lambda函数默认是一个const函数,即lambda函数的成员数据和语句操作一经定义是无法改变的。
std::vector<int> ivec{1, 2, 3};
std::vector<int> ivec1{4, 5, 6, 7, 8};
auto const_lambda = [](int a){ std::cout << a << " "; };//定义后无法修改,只能调用。
std::for_each(ivec.begin(), ivec.end(), const_lambda);
std::for_each(ivec1.begin(), ivec1.end(), const_lambda);//可以多次调用
- lambda函数是一种局部封装,局部共享,如果要全局共享,则需要使用仿函数或者一般函数。