C++11 Lambda表达式

前言


lambda表达式是C++11新特性之一,下面一起来看一下lambda表达式。


1、lambda表达式


先来看一下lambda表达式的语法形式:

[ capture ] ( params ) opt -> ret { body; };

其中carpture是捕获列表,params是参数,opt是选项,ret则是返回值的类型,body则是函数的具体实现。


1.1、carpture捕获列表描述了lambda表达式可以访问上下文中的哪些变量。


[] :表示不捕获任何变量
[=]:表示按值捕获变量
[&]:表示按引用捕获变量
[this]:值传递捕获当前的this


但是捕获列表不允许变量的重复传递:例如[=,x]
上面这种捕获是不允许的,=表示按值的方式捕获所有的变量,x相当于被重复捕获了。


1.2、params表示lambda的参数,用在{}中。


1.3、opt表示lambda的选项,例如mutable,后面会介绍一下mutable的用法。


1.4、ret表示lambda的返回值

也可以显示指明返回值,lambda会自动推断返回值,但是值得注意的是只有当lambda的表达式仅有一条return语句时,自动推断才是有效的。像下面这种的表达式就需要加上返回类型。

[](double x )->double{int y = x ;return x - y;};

2、lambda表达式简单示例

2.1、采用lambda来替换fun7函数。与fun7函数对应的lambda表达式如下:

[](int x) {return x%7==0;}


因此 ,我们可以直接使用上面的表达式直接替换fun7,代码如下:

 
  1. #include <iostream>

  2. #include <algorithm>

  3. #include <vector>

  4. #include <cmath>

  5. using namespace std;

  6. //为判断元素是否能被13整除,可以使用下列的函数:

  7. //bool fun7(int x)

  8. //{

  9. //    return x%13==0;

  10. //}

  11. int main()

  12. {

  13.     vector<int> vec(10);

  14. //generate()的用法,该函数接受一个区间,由前两个参数指定,并将区间中的每个元素设置为第三个参数返回的值

  15.     generate(vec.begin(),vec.end(),rand);

  16.     for(int i = 0;i<10;i++)

  17.         cout << vec[i] << "  ";

  18.     cout << endl;

  19. //count_if()算法和generate()函数差不都,同样前两个参数也是指定了一个区间,第三个参数则是一个bool 类型的函数对象

  20. //int count = count_if(vec.begin(),vec.end(),fun7);

  21.     int count = count_if(vec.begin(),vec.end(),[](int x){return x % 7 == 0;}   //使用lambda表达式替换fun7函数

  22.     );

  23.     cout << count << endl;

  24.     return 0;

  25. }


2.2、lambda表达式是匿名函数,实际上也可以给lambda表达式指定一个名称,如下表示:

auto f = [](int x ){return x % 3 ==0;};

此后再需要使用该lambda表达式,就可以使用f()来代替。举一个例子:

 
  1. #include <iostream>

  2. using namespace std;

  3. int main()

  4. {

  5.     int a = 5,b = 6;

  6.     auto f = [=]{return a+b;};//[=]按值捕获了a和b

  7.     cout << f() << endl;

  8.     return 0;

  9. }



运行程序输出结果为11,由上栗看出可以给lambda指定一个名称,再用该名称来取代调用该表达式。

2.3、按值捕获和按引用捕获

按值捕获和按引用捕获的用法通过下面这个例子来看一下。

 
  1. #include <iostream>

  2. using namespace std;

  3. int main()

  4. {

  5.     int a = 5;

  6.     auto f1 = [=]{return a+1;};//按值捕获a

  7.     auto f2 = [&]{return a+1;};//按引用捕获a

  8.     cout << f1() << endl;

  9.     cout << f2()<< endl;

  10.     a++;

  11.     cout << f1() << endl;

  12.     cout << f2() << endl;

  13.     return 0;

  14. }



程序运行结果如下:

先看第一次f1和f2的输出,都是6,这没有什么问题。再执行a++,输出f1和f2,f2的结果是7 也没有问题,为什么f1还是输出6呢?答案就是按值捕获可以理解为一旦lambda按值捕获某个变量相当于在表达式内部已经生成了一个被捕获变量的副本,而lambda表达式使用的就是这个副本,原本的变量再怎么变化都不会影响到副本的值,所以f1 lambda表达式中的值一直都是捕获时a 的值 也就是5,后续a++的操作和f1表达式没有关系。简单可以理解为f1表达式赋值了一个和a同名的const变量。从这我们也可以得出一个结论:


如果希望lambda函数在调用时访问的外部变量是最新的,我们就需要使用按引用捕获。

2.4、按值捕获[=],修改lambda中变量的值,关键字mutable使用示例


下面可以看一下这个例子:

 
  1. #include <iostream>

  2. using namespace std;

  3. int main()

  4. {

  5.     int a = 5;

  6.     auto f = [=]{return a*=5;};//按值捕获a

  7.     cout << f() << endl;

  8.     return 0;

  9. }


运行程序,有如下编译报错:

提示a是一个只读的,不允许修改,这就验证上面例子中说明的按值捕获实际上是lambda拷贝了一个与被捕获变量同名的const 副本并进行操作。
如果实在需要改变lambda中的值,这时就需要使用上文提到过的选项mutable。
默认情况下,lambda函数是一个const函数,而mutable也可以取消常量性。例如有如下代码:

 
  1. #include <iostream>

  2. using namespace std;

  3. int main()

  4. {

  5. int a = 5;

  6. auto f = [=]()mutable{return a*=5;};//取消常量性

  7. cout << f() << endl;

  8. return 0;

  9. }


这里需要注意,被 mutable 修饰的 lambda 表达式就算没有参数也要写明参数列表。


运行程序如下:

可以看到原先lambda函数中的a是只读的,加上mutable就可以修改lambda函数中a的值了。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值