摘要
lambda表达式
一句话笔记
- 使用
引用捕获
方式时,确保lambda表达式运行时,所引用的变量还存在
。例如lambda表达式引用捕获了函数的局部变量,当函数返回该lambda表达式时,局部变量已经释放,往后调用lambda表达式便会出错。 - 使用
值捕获
方式时,注意,lambda表达式在创建时,将捕获值拷贝到内部的一个同名变量中,并且该同名变量是const的
,后面该捕获变量的修改不影响lambda内的同名变量值。 - 使用
值捕获
方式时,lambda不能改变拷贝进来的同名变量的值,如果我们希望能改变该同名变量的值,需要加上关键字mutable
。
int v1 = 50;
auto f0 = [v1]() -> int {return v1;}; // 值捕获方式,此时v1的值已经拷贝,为50,不可修改v1
//auto f0 = [v1]() -> int {return ++v1;}; // 错误,不可修改v1
auto f1 = [v1]() mutable -> int {return ++v1;}; // 值捕获方式,此时v1的值已经拷贝,为50,mutable关键字,可修改v1
auto f2 = [&v1]() -> int {return v1;}; // 引用捕获方式
v1 = 10; // 这里修改了v1的值,但是f和f1里的v1值还是50
auto v2 = f0(); // v2==50
auto v3 = f1(); // v3==51
auto v4 = f2(); // v4==10
典型使用场景
- lambda函数相对于普通函数有一个优点,即它可以捕获外层局部变量的值,这使得在某种场合适用lambda函数更方便。例如,
find_if
只接受一元谓词(函数只有一个参数)
,然而当我们需要用到两个局部变量作为参数时,采用函数的方法只能接收一个形参,另一个参数必须得是全局变量(但我们需要的是两个局部变量),这时候lambda函数的参数捕获就派上用场了:
bool check_size(const string &s, string::size_type sz)
{
return s.size() >= sz;
}
void fun()
{
string::size_type sz = 100; //局部变量
// 方法一:普通函数的实现方式
// _1是占位符,表示check_my函数的第一个参数放到_1处,
// 然后std::bind再把_1, sz两个参数依次传递给函数check_size,即调用check_size(_1, sz)。
auto check_my = std::bind(check_size, _1, sz);
fin_if(str.begin(), str.end(), check_my);
// 方法二:普通函数的实现方式
fin_if(str.begin(), str.end(), std::bind(check_size, _1, sz));
// 方法三:lambda表达式的实现方式
fin_if(str.begin(), str.end(), [sz](const string &s){
return s.size() >= sz; //sz是lambda捕获的局部变量,而用普通函数难以获得
});
}
相关知识点
- 参数绑定:std::bind, std::ref, std::cref
相关/参考链接
《C++ prime 第五版》p344-357