1、匿名函数的使用
匿名函数的基本语法为:
//[捕获列表](参数列表)->返回类型{函数体}
int main()
{
auto Add = [](int a, int b)->int {
return a + b;
};
std::cout << Add(1, 2) << std::endl;
return 0;
}
上述代码便会输出3,这就是匿名函数的使用方法。
注意点:一般情况下,编译器可以自动推断出lambda表达式的返回类型,所以我们可以不指定返回类型,即:
//[捕获列表](参数列表){函数体}
int main()
{
auto Add = [](int a, int b) {
return a + b;
};
std::cout << Add(1, 2) << std::endl;
return 0;
}
但是如果函数体内有多个return语句时,编译器无法自动推断出返回类型,此时必须指定返回类型
2、什么是捕获列表?
捕获列表:lambda表达式的捕获列表精细控制了lambda表达式能够访问的外部变量,以及如何访问这些变量。
试图在Lambda内使用外部变量是错误的,例如:
#include <iostream>
int main()
{
int c = 12;
auto Add = [](int a, int b)->int {
return c;
};
std::cout << Add(1, 2) << std::endl;
return 0;
}
但是有些时候我们需要使用外部变量,所以需要使用捕获列表,上述代码改写成一下形式便可以通过编译。
#include <iostream>
int main()
{
int c = 12;
auto Add = [c](int a, int b)->int {
return c;
};
std::cout << Add(1, 2) << std::endl;
return 0;
}
但是如果你更改一下lambda表达式中代码块的内容,比如在上述代码中添加一句
c = a;
这样又会无法编译通过了,因为你的c是按值传递的,所以要将捕获列表改成
[&c](int a, int b)->int{……};
这样c的值便是按引用传递了,便可以进行修改。
补充知识:
如果捕获列表为[&],则表示所有的外部变量都按引用- 传递给lambda使用
如果捕获列表为[=],则表示所有的外部变量都按值传递给lambda使用
匿名函数构建的时候对于按值传递的捕获列表,会立即将当前可以取到的值拷贝一份作为常数,然后将该常数作为参数传递,即:
int main()
{
int c = 12;
//Add构建的时候实际是:[12](int a,int b)->int{}
auto Add = [c](int a, int b)->int {
return c;
};
std::cout << Add(1, 2) << std::endl;
return 0;
}
到此,匿名函数(Lambda表达式)的基础使用方法就介绍完毕了。
3、附加
3.1、介绍Lambda
一个Lambda表达式表示一个可调用的代码单元,我们可以将它理解为一个未命名的内联函数。和任何函数类似,一个Lambda表达式具有一个返回类型,一个参数列表和一个函数体,但和普通函数不一样的是Lambda表达式可能定义在函数内部。我们可以忽略Lambda表达式的参数列表和返回类型,但不可以忽略捕获列表和函数体。
//这是正确的(忽略了参数列表和返回类型)
//为什么没有指定返回类型还可以返回整数值?
//因为你的函数体有return语句,它可以推断出来
auto f = []{ return 1 + 2; };
3.2、Lambda捕获列表
[] 不捕获任何变量。
[&] 捕获外部作用域中所有变量,并作为引用在函数体中使用(按引用捕获)。
[=] 捕获外部作用域中所有变量,并作为副本在函数体中使用(按值捕获)。注意值捕获的前提是变量可以拷贝,且被捕获的变量在 lambda 表达式被创建时拷贝,而非调用时才拷贝。如果希望lambda表达式在调用时能即时访问外部变量,我们应当使用引用方式捕获。
[=,&foo]按值捕获外部作用域中所有变量,并按引用捕获foo变量。
[bar]按值捕获bar变量,同时不捕获其他变量。
[this]捕获当前类中的this指针,让lambda表达式拥有和当前类成员函数同样的访问权限。如果已经使用了&或者=,就默认添加此选项。捕获this的目的是可以在lamda中使用当前类的成员函数和成员变量。
参考资料:《C++ Primer(第五版)》
转载:https://www.cnblogs.com/yaya12138/p/11815475.html