提示:文章
前言
前期疑问:
本文目标:
lamda表达式
一、背景
看c++科二的时候有看到lamda表达式,就再次看了下
这次稍微有点看懂
lamda表达式结构:
lambda的结构是:[ ] () -> 返回值类型 { }。从左到右依次是:捕捉列表 函数参数 -> 返回值类型 函数体。
解析上述的结构
在使用lambda的时候,有几个参数是可以直接省略的:
1.参数列表可以省略(写() 或者不写())。
2.返回值可以省略(不写 -> 类型)。
除此之外的东西都不可省略(捕捉列表,函数体实现)。
上述变化最多的应该就是捕捉列表。
[ ] | 不捕捉任何变量 | |
---|---|---|
[var] | 表示值传递方式捕捉变量var | 显式捕捉 |
[=] | 表示值传递方式捕获所有父作用域中的变量(包括this) | 隐式捕捉 |
[&var] | 表示引用传递捕捉变量var; | 显式捕捉;可以改变变量值 |
[&] | 表示引用传递捕捉所有父作用域中的变量(包括this); | 隐式捕捉;可以改变变量值 |
[this] | 捕获当前类中的 this 指针 |
a. 父作用域指包含lambda函数的语句块
b. 语法上捕捉列表可由多个捕捉项组成,并以逗号分割。
比如:[=, &a, &b]:以引用传递的方式捕捉变量a和b,值传递方式捕捉其他所有变量 [&,a, this]:值传递方式捕捉变量a和this,引用方式捕捉其他变量
auto fun1 = [&](int c) {
b = a + c; };
fun1(10);
cout << a << " " << b << endl;
c. 捕捉列表不允许变量重复传递,否则就会导致编译错误。 比如:[=, a]:=已经以值传递方式捕捉了所有变量,捕捉a重复
int x = 10;
auto add_x = [x, =](int a) mutable {
x *= 2; return a + x; };
cout << add_x(10) << endl;
d. 在块作用域以外的lambda函数捕捉列表必须为空。
e. 在块作用域中的lambda函数仅能捕捉父作用域中局部变量,捕捉任何非此作用域或者非局部变量都会导致编译报错。
f. lambda表达式之间不能相互赋值,即使看起来类型相同
lambda表达式实际上可以理解为无名函数,该函数无法直接调用,如果想要直接调用,可借助auto将其赋值给一个变量。
二、题目
下面加上喂饭宝典的几个题目。
1、(单选)【lambda】下面代码运行结果是(B)
int n = 1;
auto func1 = [=] {
return n; };
auto func2 = [&] {
return n; };
n++;
cout<<func1()<<" "<<func2()<<endl;
A. 1 1
B. 1 2
C. 2 2
D. 2