浅谈C++11新引入的lambda表达式

 更新时间:2017年07月29日 12:02:11   作者:lenomirei  
Lambda表达式(又称Lambda函数,英文原文是Lambda Expression),是C++11的新特性中非常实用的一个。

ISO C++ 11 标准的一大亮点是引入Lambda表达式。基本语法如下:

[capture list] (parameter list) ->return type { function body }

简单的讲一下各个部分的作用

1.[capture list]捕获列表,捕获到函数体中,使得函数体可以访问
2.(parameter list)参数列表,用来表示lambda表达式的参数列表
3.->return type函数返回值 {function body}就是函数体

lambda表达式可以理解为一个匿名函数(但本质并不是),如果要使用lambda表达式声明的函数,需要给他“命名”

lambda表达式可以表示闭包,因为本身就是这个类
闭包是指可以包含自由变量的代码块 (未绑定到特定的对象:举个栗子std::function就可以产生一个对象,或者一个未指向任何函数的函数指针)

闭包讲的通俗一些有以下几点

1.自带上下文的函数,闭包可以储存运行时需要的上下文,这样就可以在上下文不存在的时候还可以使用闭包(变量a生命周期到了被销毁,但是在闭包中还可以拿来用)
2.可以把闭包看成一个重载了operator()的类,带有状态的意思就可以解释成通过this指针使用成员变量
3.capture list就是lambda表达式实现闭包的方式

简单使用的例子

--------------------------------------------------------------------------------

C++11为auto提供了新的功能,如他的名字一般,现在可以看成自动适应类型,可以适应多数类型
使用auto来代替变量的类型,前提是被明确类型的初始化变量初始化的,可以使用auto关键字

 auto f = [](){}; 
 auto f = [](int a, int b)->int {return a + b; };
 f(1, 2);//需要这么使用

只要是函数类型就都可以使用这个lambda表达式

 typedef int(*FUNC)(int a, int b);
 int main()
 {
   FUNC a= [](int a, int b) {return a + b; };
  
   printf("%d\n", a(1, 2));
 }

声明函数的方法都可以接收不带捕获列表的lambda表达式

 typedef std::function<int(int a, int b)> FUNC;
 int main()
 {
   FUNC a= [](int a, int b) {return a + b; };
  
   printf("%d\n", a(1, 2));
 }

--------------------------------------------------------------------------------

lambda表达式中capture list的用法

 int func(int a, int b, std::function<int(int, int)> f)
 {
   return f(a, b);
 }
 
 
 int a=1;
 int b=2;
 int c=3;
 int d = func(a, b, [a, &b](int m, int n) {
 
     printf("a = %d\n", a); // a是通过值传递捕获,mutable只在函数体内修改有效
     printf("b = %d\n", b); // b是引用传递捕获,mutable可以对外部b造成影响
 
                //printf("c = %d\n", c); // c不可访问
 
     return m + n;
   });
 typedef int(*FUNC)(int m, int n,std::function<int(int ,int )> f);
 
 void test()
 {
   FUNC oho;
   int a = 10;
   int b = 20;
   auto func = [&a, &b](int m, int n) {printf("a:%d b:%d\n", a, b); return m + n; };
   
 }

1.[]空。没有使用任何函数对象参数。
2.[=]。函数体内可以使用Lambda所在作用范围内所有可见的局部变量(包括Lambda所在类的this),并且是值传递方式(相当于编译器自动为我们按值传递了所有局部变量)。
3.[&]。函数体内可以使用Lambda所在作用范围内所有可见的局部变量(包括Lambda所在类的this),并且是引用传递方式(相当于编译器自动为我们按引用传递了所有局部变量)。
4.[this]。函数体内可以使用Lambda所在类中的成员变量。
5.[a]。将a按值进行传递。按值进行传递时,函数体内不能修改传递进来的a的拷贝,因为默认情况下函数是const的。要修改传递进来的a的拷贝,可以添加mutable修饰符。
6.[&a]。将a按引用进行传递。
7.[a, &b]。将a按值进行传递,b按引用进行传递。
8.[=,&a, &b]。除a和b按引用进行传递外,其他参数都按值进行传递。注意=符号的位置必须在头一个
9.[&, a, b]。除a和b按值进行传递外,其他参数都按引用进行传递。&符号的位置必须在头一个

当你想改变通过传值方式捕捉的变量的时候就要添加mutable

[a, &b, &b2](int m, int n)mutable {a *= 2; return m*n; }:

--------------------------------------------------------------------------------

lambda表达式的其他用法

 class A
 {
 public:
   A();
   ~A();
   void test()
   {
     auto f = [this](int m, int n) {printf("%d\n", a); };
   }
     
 private:
   int a;
 };

lambda表达式本质是一种闭包类型,虽然他可以赋值给函数指针,但是只限于在捕获列表为空的时候,当捕获列表有值的时候,应该使用auto来接收lambda表达式,或者用std::function也是可以的

 main::__l2::<lambda_eb7b0a89c14bee3d2620c108ffb635c6>
 //这是一个lambda表达式在VS2015环境下显示的类型,不用auto用什么来接收调用他呢?

本质来说lambda表达式之间是不允许赋值的

 auto a = [](int m, int n) {return m + n; };
 auto b = [](int m, int n) {return m - n; };
 a = b;

操作非法,因为闭包类型不允许使用赋值操作符,但是函数指针可以,也就是可以有下面的操作

 typedef int(*FUNC)(int a, int b);
 int main()
 {
   FUNC a = [](int a, int b) {return a + b; };
   FUNC b = [](int a, int b) {return a + b; };
   a = b;
   return 0;
 }

std::function之间也是可以赋值的,他就可以办到有capture list的lambda表达式进行赋值

 typedef std::function<int(int,int)> FUNC;
 int m = 10;
 int n = 20;
 FUNC a = [m, n](int a, int b){printf("%d\n", m); return a + b; };
 FUNC b = [m, n](int a, int b){return a + b; };
 b = a;
 b(1, 2);
 //执行结果是可以把m打印出来的

以上就是本文给大家介绍的c++11的新特性lambda表达式的全部内容了,希望大家能够喜欢

相关文章

  • c语言实现奇偶排序算法

    c语言实现奇偶排序算法

    这篇文章主要介绍了c语言实现奇偶排序算法,有需要的朋友可以参考一下
    2013-12-12
  • socket编程之bind()函数使用示例详解

    socket编程之bind()函数使用示例详解

    这篇文章主要为大家介绍了socket编程之bind()函数使用示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-10-10
  • Qt5连接并操作PostgreSQL数据库的实现示例

    Qt5连接并操作PostgreSQL数据库的实现示例

    本文主要介绍了Qt5连接并操作PostgreSQL数据库的实现示例,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-12-12
  • C语言位图及位图的实现

    C语言位图及位图的实现

    这篇文章主要为大家详细介绍了C语言位图及位图的实现,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-05-05
  • opencv3/C++视频中叠加透明图片的实现

    opencv3/C++视频中叠加透明图片的实现

    今天小编就为大家分享一篇opencv3/C++视频中叠加透明图片的实现,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-12-12
  • VC定制个性化的MessageBox解决方法

    VC定制个性化的MessageBox解决方法

    这篇文章主要介绍了VC定制个性化的MessageBox解决方法,有助于进一步的了解windows应用程序的消息机制及运行原理,需要的朋友可以参考下
    2014-07-07
  • C语言 超详细梳理总结动态内存管理

    C语言 超详细梳理总结动态内存管理

    动态内存是相对静态内存而言的。所谓动态和静态就是指内存的分配方式。动态内存是指在堆上分配的内存,而静态内存是指在栈上分配的内存,本文带你深入探究C语言中动态内存的管理
    2022-03-03
  • C语言中结构体、联合体的成员内存对齐情况

    C语言中结构体、联合体的成员内存对齐情况

    这篇文章主要给大家介绍了关于C语言中结构体、联合体的成员内存对齐情况的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-05-05
  • C++异步调用方法详解

    C++异步调用方法详解

    这篇文章主要介绍了C++异步调用方法,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-05-05
  • 详解C语言的基本数据类型

    详解C语言的基本数据类型

    这篇文章主要为大家介绍了C语言的基本数据类型,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-01-01

最新评论