说明一下,我用的是gcc7.1.0编译器,标准库源代码也是这个版本的。
本篇文章讲解c++11中lambda表达式用法。
初次接触lambda这个关键字,记得还是在python里面,但其实,早在2011年c++11推出来的时候我们c++就有了这个关键字啦。lambda表达式是C++11中引入的一项新技术,利用lambda表达式可以编写内嵌的匿名函数,用以替换独立函数或者函数对象,并且使代码更可读。
所谓函数对象,其实就是对operator()进行重载进而产生的一种行为,比如,我们可以在类中,重载函数调用运算符(),此时类对象就可以直接类似函数一样,直接使用()来传递参数,这种行为就叫做函数对象,同样的,它也叫做仿函数。
如果从广义上说,lambda表达式产生的是也是一种函数对象,因为它也是直接使用()来传递参数进行调用的。
1 lambda表达式基本使用
lambda表达式基本语法如下:
[ 捕获 ] ( 形参 ) -> ret { 函数体 };
lambda表达式一般都是以方括号[]开头,有参数就使用(),无参就直接省略()即可,最后结束于{},其中的ret表示返回类型。
我们先看一个简单的例子,定义一个可以输出字符串的lambda表达式,完整的代码如下:
#include <iostream>
int main()
{
auto atLambda = [] {
std::cout << "hello world" << std::endl;};
atLambda();
return 0;
}
上面定义了一个最简单的lambda表达式,没有参数。如果需要参数,那么就要像函数那样,放在圆括号里面,如果有返回值,返回类型则要放在->后面,也就是尾随返回类型,当然你也可以忽略返回类型,lambda会帮你自动推导出返回类型,下面看一个较为复杂的例子:
#include <iostream>
int main()
{
auto print = [](int s) {
std::cout << "value is " << s << std::endl;};
auto lambAdd = [](int a, int b) ->int {
return a + b;};
int iSum = lambAdd(10, 11);
print(iSum);
return 0;
}
lambAdd有两个入参a和b,然后它的返回类型是int,我们可以试一下把->int
去掉,结果是一样的。
2 lambda捕获块
2.1 捕获的简单使用
在第1节中,我们展示了lambda的语法形式,后面的形参和函数体之类都好理解,那么方括号里面捕获是啥意思呢?
其实这里涉及到lambda表达式一个重要的概念,就是闭包。
这里我们需要先对lambda表达式的实现原理做一下说明:当我们定义一个lambda表达式后,编译器会自动生成一个匿名类,这个类里面会默认实现一个public类型的operator()函数,我们称为闭包类型。那么在运行时,这个lambda表达式就会返回一个匿名的闭包实例,它是一个右值。
所以,我们上面的lambda表达式的结果就是一个一个的闭包。闭包的一个强大之处是可以通过传值或者引用的方式捕获其封装作用域内的变量,前面的方括号就是用来定义捕获模式以及变量,所以我们把方括号[]括起来的部分称为捕获块。
看这个例子:
#include <iostream>
int main()
{
int x = 10;
auto print = [](int s) {
std::cout << "value is " << s << std::endl;