目录
1. 什么是lambda表达式
其本质就是一个匿名函数对象,在编译器看来就是一个仿函数。lambda表达式可以在定义时直接嵌入代码,无需单独定义函数名称、参数以及返回值。
2. 为什么用lambda表达式
传统的函数对象的使用(比如sort(arr.begin(), arr.end(), cmpLess()); // 升序),需要先定义一个类,再重载()运算符,再将function()作为参数传入【也就是仿函数的用法】。
lambda表达式可以在使用的时候方便的定义函数对象或者回调函数,可以直接在传参时直接编写函数对象的代码逻辑(比如sort(arr.begin(), arr.end(), [](int n1, int n2) { return n1< n2; }); // 升序),并且一定程度上提高代码的可阅读性。
其优点:简洁、高效、更加灵活。
3. lambda表达式的语法
lambda表达式:[ ]( ) mutable ->returntype { }
[ ]
捕捉列表( )
参数列表mutable
关键字->returntype
返回值类型{ }
函数体
(1)捕获列表[ ]:将当前作用域中的变量以值传递或者引用传递的当时送入函数使用。【可以不用传入参数,而直接捕获】
1.传值捕获:
auto swap = [x, y]()mutable ->void
{
auto tmp = x;
x = y;
y = tmp;
}
2.引用捕获:
auto swap = [&x, &y]() ->void
{
auto tmp = x;
x = y;
y = tmp;
};
3.混合捕获:
auto func = [&x, y]()mutable ->void
{
x = 100;
y = 200;
};
4.全部引用捕获:
auto func = [&]()->void
{
cout << x << " " << y << " " << z << " " << endl;
cout << a << " " << b << " " << c << " " << endl;
cout << str << endl;
cout << "&str: " << &str << endl << endl;
};
5.全部传值捕获:
auto func = [=]()->void
{
cout << x << " " << y << " " << z << " " << endl;
cout << a << " " << b << " " << c << " " << endl;
cout << str << endl;
cout << "&str: " << &str << endl << endl;
};
(2)参数列表()、关键字、返回值类型:这三者可以省略,其中,参数列表按照需要是否传参,返回值类型编译器可以推导。
(3)函数体{ }:可以使用捕获的外部变量,也可以使用传入的参数,实现对应的逻辑。
4. lambda表达式的用法
lambda
表达式 构建出的是一个 匿名函数对象,匿名函数对象也可以调用,不过需要在创建后立即调用,否则就会因为越出作用域而被销毁(匿名对象生命周期只有一行) 。
int main()
{
int ret = [](int x, int y)->int { return x + y; }(1, 2);
cout << ret << endl;
return 0;
}
5. lambda表达式的原理
lambda
表达式 生成的函数对象与 仿函数 生成的函数对象大小是一样的,都是 1
字节。
- 仿函数 生成的函数对象大小为
1
字节是因为其生成了一个空类,实际调用时是通过operator()
重载实现的,比如上面的addFunc
类,空类因为没有成员变量,所以大小只为1
字节。 lambda
表达式 本质上也是生成了一个空类
所以在编译器看来,lambda
表达式 本质上就是一个 仿函数