1.基本
这个好像是很好用的,其有以下有点:
- 声明式的编程风格:直接匿名定义目标函数或函数对象,不需要额外写一个命名函数或函数对象。
- 简洁:避免了代码膨胀和功能分散,让开发更加高效。
- 在需要的时间和地点实现功能闭包,使程序更加灵活。
书写格式如下:
[capture](params)opt ->ret{body;};
capture是捕获l列表,params是参数列表,opt是函数选项,ret是返回值,body是函数体本身。
2.capture捕获列表
lambda表达式的捕获列表可以捕获一范围内的变量,具体使用方式如下:
- []-不捕获任何变量
- [&]-捕获外部作用域中所有变量,并作为引用在函数体内使用(按引用捕获)
- [=]-捕获外部作用域中所有变量,并作为副本在函数体内使用(按值捕获)(拷贝的副本在匿名函数体内部是只读的,要想修改其值需要加上函数选项 mutable)
- [=, &foo]-按值捕获外部作用域中所有变量,并按照引用捕获外部变量foo
- [bar]-按值捕获bar变量,同时不捕获其他变量
- [&bar]-按引用捕获bar变量,同时不捕获其他变量
- [this]-捕获当前类中的this指针 (让lambda表示拥有和当前类成员函数同样的访问权限,如果已经使用了&或者=,迷人添加此选项。)
一些示例:
注:在匿名函数内部,需要通过lambda表示的捕获刘表控制如何捕捉外部变量,以及访问那些变量,以及访问那些变量。默认的访问状态下lambda表达式无法修改通过赋值方式捕捉外部变量,如果希望改变这些外部变量,需要通过引用的方式进行捕获,或者加上函数选项mutable
3.返回值
很多时候,lambda表达式返回值是非常明显的,因此C++11中允许省略表达式的返回值
注:一般情况下lambda都可以自动推导,但是注意的是lambda表达式不能通过列表初始化自动推导出返回值类型。
4.函数本质
在上述提到使用=捕捉外部值得时候不可以修改捕捉得值原因是:
- lambda表达式得类型在c++11中会被看做成一个带operator得类,即仿函数。
- 按照C++标准,lambda表达式得operator()默认得是const的,一个const成员无法修改成员变量值的。
因为lambda表达式在c++中会被看成一个仿函数,因此可以使用std::function和std::bind来存储和操作lambda表达式:
补充:对于没有任何捕捉变量的lambda表达式可以看是一个普通的函数指针:
--------------------------------------------------
上述如有错误欢迎大家指正。