1、什么是lambda演算
关于这个问题,可以看:点击打开链接(songwang的回答,一篇篇看下来对初学者很有帮助!)。如果想节省时间可以看:这个。
2、C++的lambda简介
C++中的lambda可以看成一个有自己的格式和使用规则的函数。
lambda表达式的格式是:[/*capture list*/](data_type para1,data_type para2...)->return_type{/*function body*/}
第一个[]叫做捕获列表,用来捕获外界值,()里写函数参数,->后写函数返回类型(可以省略),{}中是函数体。看一些例子:
auto f1=[]{return 42;};// 定义一个lambda,不接受参数直接返回42,f1是一个函数
auto f2=[]()->int{return 42;};// 同上
//cout<<f2()<<endl;
vector<int> vec1{1,2,3};
int offset=2;
for_each(vec1.begin(),vec1.end(),[offset](int& num){num+=offset;});// 捕获上面的offset,每个元素都加offset
//for(auto i:vec1)cout<<i<<" ";
auto sum=[](int a,int b){return a+b;};// lambda实现两个数相加
//cout<<sum(1,2)<<endl;
lambda作为一个可调用对象,和C++的algorithm库简直是绝配,合理使用lambda会让我们的代码更简洁、有力。
3、lambda的捕获和返回
(1)值捕获
值捕获的前提是变量可拷贝,拷贝发生在lambda创建时:
注意,一般的值捕获不能修改被捕获的值。例如,假设在上面的auto f=[v1]{return v1;}中进行了一些修改v1的操作,编译器会报错。
(2)引用捕获
就是在[]里写一个引用。
int num1=1;
auto inc=[&num1]{num1++;};
cout<<num1<<endl;
注意上面例子打印出来的还是1,因为不管是值捕获还是引用捕获,lambda捕获的都是局部变量,一旦lambda执行结束,变量就不复存在了,在这个例子中,在lambda执行结束后,捕获的引用指向的局部变量已经消失了,所以并不影响原来的num1
(3)隐式捕获
使用&表示隐式的引用捕获,=表示隐式的值捕获。如果隐式和显式混用,&或者=必须写在捕获列表第一个。例子:
int num1=1,num2=2,num3=3;
auto sum_three=[=]{return num1+num2+num3;};// 隐式值捕获
cout<<sum_three()<<endl;// 打印6
(4)改变捕获的值
对于值捕获,有时需要改变捕获的值,可以使用mutable:
注意这个说的“改变捕获的值”是说允许在lambda内部修改这个值了,但是修改后的结果和捕获值本身无关,例如:
int v1=20;
auto plus_six=[v1]mutable(){v1+=6;return v1;};
auto v2=plus_six();
cout<<v1<<" "<<v2<<" "<<endl;
打印的是20 26,而不是说v1也变成了26。
(5)返回类型: