在C++11以后,形如这样的语法。 [capture](parameters)mutable ->return-type{statment}称为一个lambda表达式,表达一个匿名函数。从编程的角度上看,lambda是一种函数式编程。
分析语法: 1.capture,可以通过值传递或者引用传递的方式给匿名函数提供访问父作用域上下文的变量。特殊的两种情况:[=]和[&]分别表示,以值传递的形式捕捉所有父作用域的变量和以引用传递的形式捕捉所有父作用域的变量。 2.()参数列表形如一般的函数的参数列表。没有参数时可以忽略不写。 3.可选择项,mutable,->return type,分别表示去除匿名函数的常量性,->return type是匿名函数的返回值类型,可以忽略由编译器自动推导。添加可选项必须在此之前增加()。 4.{statment}函数本体。
#include <thread>
#include <iostream>
int main(){
[]{};//最简单的表达方式
int a=2,b=2;
[=]()->int{return a+b;};
[a](){cout<<a<<endl;}; //[var] 以传值的方式捕获
[&b](){++b;cout<<b<<endl;}// [&b] 以传递引用的方式捕获
//可以使用匿名函数与线程结合起来
thread t1([]{cout<<"hello world!"<<endl;});
t1.join();
}
使用mutable关键字的区别
#include <iostream>
using namespace std;
int main(){
int id = 10;
auto f = [id]()mutable{
++ id;
cout<<"id:"<<id<<endl;
};
f();
cout<<"main id:"<<id<<endl;
f();
return 0;
}
(1)增加mutable关键字
(2)不增加mutable关键字
#include <iostream>
using namespace std;
int main(){
int id = 10;
auto f = [id]{++id;cout<<"id:"<<id<<endl;};
f();
cout<<"main id:"<<id<<endl;
f();
return 0;
}
匿名函数与仿函数有着本质的相似处
#include <iostream>
using namespace std;
//对()进行重载的函数,在调用()时与调用函数有着相似之处,但是是通过对象名称调用的
class Functor{
public:
Functor(int a=0,int b=0):_a(a),_b(b){}
void operato()(){cout<<_a+_b<<endl;}
private:
int _a;
int _b;
};
int main(){
Functor sum(10,20);
sum();
auto sum1 = [](int a,int b){cout<<a+b<<endl;};
sum1(20,20);
return 0;
}
通过上述例子,除去在语法层面上的不同,仿函数和lambda表达式都是通过捕捉一些变量作为初始值,之后进行传递参数并进行计算。在形式上,仿函数通过构造函数构造进行初始化,而lambda表达通过捕捉列表[capture]获取初始值。仿函数通过对operator()重载函数的参数列表进行传参,而lambda表达式通过(parameters)参数列表进行传递参数。最后,仿函数的执行体在operator()函数体中进行运算,而lambda表达式在{}进行运算。