本章主要内容:
一,lambda表达式
1.基本概念
2.关于捕获子句
3.常见的捕获方式
二,闭包与std::bind模板
1.什么是闭包
2.std::bind的简介
3.std::bind的用法
三,参考阅读
一,lambda表达式
1.基本概念
lambda表达式是从C 11开始引入的,主要用来定义匿名函数和闭包。lambda表达式可以被当作一个值赋给另一个变量,也可以作为实参传递给其他函数,或者作为其他函数的返回结果,用法类似于前面提到的函数对象和函数指针。如果只是把单个函数拿来传参,lambda表达式的使用方式比函数指针和函数对象更简洁。
lambda表达式可以不指定函数的返回类型,编译器将自动推导该类型。
lambda表达式的完整公式:
代码语言:javascript复制[capture_list](parameter_list) mutable -> return_type{ process code };
具体含义:
[]: lambda表达式的引出符,编译器根据该符号判断接下来的代码是否为lambda匿名函数。
parameter_list: 参数列表,与普通函数的参数列表一致。如果不需要传递参数,则可以省略该部分以及小括号()。
mutable: 使用了mutable修饰符的lambda表达式,不可以省略参数列表。
return_type: 函数返回值类型。该部分可以连同"->"一起省略。
process code: 函数体,它除了可以使用参数之外,还可以使用捕获到的变量。
lambda表达式样例:
代码语言:javascript复制[](int x, int y){return x<y;} //[]用来标记lambda表达式的开始
[](int x=0, int y=0){return x<y;} //传默认实参x=0,C 14标准开始支持
[]{return true;} //没有参数时,可以省略圆括号()
[](int x, int y)->bool{return x<y;} //显式指定返回值类型,让代码更清晰
注意,lambda表达式中的"[ ]"不一定是空的,里面可以包含捕获子句,捕获子句用来捕获上下文中的变量来提供给lambda表达式使用。
C 代码样例:
Demo1:
代码语言:javascript复制#include<iostream>
using namespace std;
int main() {
auto operation = [](int a, int b, string op) -> double {
if (op == "sum") {
return a b;
}
else {
return (a b) / 2.0;
}
};
int num1 = 1;
int num2 = 2;
auto sum = operation(num1, num2, "sum");
cout << "Sum = " << sum << endl;
auto avg = operation(num1, num2, "avg");
cout << "Average = " << avg;
return 0;
}
运行结果:
代码语言:javascript复制Sum = 3
Average = 1.5
Demo2: lambda与std::for_each结合使用
代码语言:javascript复制#include <bits/stdc .h>
#include <iostream>
using namespace std;
int main()
{
vector<int> vec{ 1, 2, 3, 4, 5 };
for_each(vec.begin(),
vec.end(),
[](int& a) { a *= 2; }
);
for_each(vec.begin(),
vec.end(),
[](int a) { cout << a << " " ; }
);
cout << endl;
return 0;
}
运行结果:
代码语言:javascript复制2 4 6 8 10
2.关于捕获子句
捕获子句定义了lambda表达式访问(捕获)表达式以外的参数和变量的方式。
默认的捕获子句有两种即"="(按值捕获)和"&"(按引用捕获)。
为什么要有捕获子句:
当[ ]中为空时,lambda表达式只能访问lambda表达式中定义的局部实参和局部变量。当[ ]中不为空时,lambda表达式可以访问代码指定作用域中的所有参数和变量。因此,捕获子句的使用扩大了lambda表达式捕获变量的范围。
3.常见的捕获方式
方式一,按值捕获
方括号中包含"=",指定作用域中变量的值可以传递到lambda表达式,lambda表达式可以使用变量的值,但是不能修改变量的值。
方式二,按引用捕获
方括号中包含"&",指定作用域中变量的引用可以传递到lambda表达式,lambda表达式既可以使用变量的值,也可以修改变量的值。
方式三,捕获指定的变量
捕获变量和默认捕获子句的操作有些区别:
按值捕获变量:[ ]中直接传变量名,不带"="。
按引用捕获变量:[ ]中传的是 "&"后面加变量名。
捕获多个变量时可以用逗号分隔,例如:
代码语言:javascript复制[=, &counter] //按引用捕获counter,按值捕获其他变量
[&, counter] //按值捕获counter,按引用捕获其他变量
指定的默认子句("="或"&")必须是捕获列表中的第一项。
如果捕获列表的前面已经加了"="捕获子句,则后面不能再按值捕获特定变量。同理,如果捕获列表的前面已经加了"&"捕获子句,则后面不能再按引用捕获特定变量。
所以下面这两个捕获子句会产生编译错误:
代码语言:javascript复制[&, &counter]
[=, &counter, number]
方式四,捕获this指针
如果一个对象的成员函数中有lambda表达式,那么这个lambda表达式不能通过按值捕获或按引用捕获这个对象的成员变量。为了让lambda表达式能够访问当前对象的成员变量,应该在捕获子句中使用this关键字。
有了this指针,lambda表达式可以访问当前对象的所有成员函数和成员变量,无论它们的访问权限被声明为protected还是private。
总结下来,常见的捕获语法有:
[=]: 按值捕获所有变量。
[&]: 按引用捕获所有变量。
[=,&x,&y]: 按引用捕获变量x和y,按值捕获其他变量。
[&,x,y]: 按值捕获变量x和y,按引用捕获其他变量。
[this]: 捕获当前的对象。
[*this]: 捕获当前的对象的副本。
C 代码样例
Demo1:
代码语言:javascript复制#include<iostream>
using namespace std;
int main() {
int initial_sum = 100;
auto add_to_sum = [initial_sum](int num) {
return initial_sum num;
};
int final_sum = add_to_sum(78);
cout << "100 78 = " << final_sum;
return 0;
}
运行结果:
代码语言:javascript复制100 78 = 178
Demo2:
代码语言:javascript复制#include <functional>
#include <iostream>
using namespace std;
int main()
{
int i = 3;
int j = 5;
function<int(void)> f1 = [i, j] { return i j; };
i = 22;
j = 44;
function<int(void)> f2 = [&i, &j] { return i j; };
cout << f1() << endl;
cout << f2() << endl;
}
运行结果:
代码语言:javascript复制8
66
Demo3:
代码语言:javascript复制#include <algorithm>
#include <iostream>
#include <vector>
using namespace std;
class Scale
{
public:
explicit Scale(int scale) : _scale(scale) {}
void ApplyScale(const vector<int>& v) const
{
for_each(v.begin(), v.end(), [this](int n) { cout << n * _scale <<" "; });
cout << endl;
}
private:
int _scale;
};
int main()
{
vector<int> values;
values.push_back(1);
values.push_back(2);
values.push_back(3);
values.push_back(4);
Scale s(3);
s.ApplyScale(values);
}
运行结果:
代码语言:javascript复制3 6 9 12
二,闭包与std::bind模板
1.什么是闭包
闭包( Closure)这个概念起源于函数式编程,是指外部变量与函数之间的绑定,可以这样理解,捕获了外部变量的lambda表达式是一种闭包。
2.std::bind的简介
std::bind是C 11标准引入的函数模板,用于取代bind1st和bind2nd等旧式语法。std::bind常用来实现闭包,
它用于包装和调用特征相同的函数指针、函数对象或lambda表达式。
std::bind可以充当函数适配器,即它接受一个原函数作为输入并返回一个新的函数对象作为输出,返回的函数对象包含一个或多个与原函数绑定的参数。std::bind可以预先指定函数的所有参数,也可以将函数的部分参数预先指定好,剩下的参数等真正调用的时候再指定。
3.std::bind的用法
假如有一个计算两个数字相加的函数。
代码语言:javascript复制int add(int first, int second)
{
return first second;
}
std::bind将函数名作为其第一个参数,后面的参数用"_1,_2"这样的占位符来预留,得到一个函数对象add_func。add_func可以像函数一样直接被调用。
代码语言:javascript复制auto add_func = std::bind(&add, _1, _2);
add_func(4,5); //4 5, 返回9
假设遇到了特殊场景,需要将函数的第一个参数传12,第二个参数作为预留,使用方式如下。
代码语言:javascript复制auto new_add_func = std::bind(&add, 12, _1);
new_add_func(5); //12 5, 返回17
完整C 代码实现:
代码语言:javascript复制#include <iostream>
#include <functional>
using namespace std;
int add(int first, int second)
{
return first second;
}
int main()
{
auto new_add_func = std::bind(&add, 12, placeholders::_1);
int x = new_add_func(5);
cout << x << endl;
return 0;
}
运行结果:
代码语言:javascript复制17