1. lambda表达式

//C++ 11中的Lambda表达式用于定义并创建匿名的函数对象,以简化编程工作。Lambda的语法形式如下:

//[函数对象参数](操作符重载函数参数)mutable或exception声明->返回值类型{函数体 }

//可以看到,Lambda主要分为五个部分:

//[函数对象参数]、(操作符重载函数参数)、mutable或exception声明、->返回值类型、{函数体 }。下面分别进行介绍。

//一、[函数对象参数],标识一个Lambda的开始,这部分必须存在,不能省略。函数对象参数是传递给编译器自动生成的函数对象类的构造函数的。

//函数对象参数只能使用那些到定义Lambda为止时Lambda所在作用范围内可见的局部变量(包括Lambda所在类的this)。函数对象参数有以下形式:

//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按值进行传递外,其他参数都按引用进行传递。

//二、(操作符重载函数参数),标识重载的()操作符的参数,没有参数时,这部分可以省略。参数可以通过按值(如:(a, b))和按引用(如:(&a, &b))两种方式进行传递。

//三、mutable或exception声明,这部分可以省略。按值传递函数对象参数时,加上mutable修饰符后,可以修改按值传递进来的拷贝(注意是能修改拷贝,而不是值本身)。

//exception声明用于指定函数抛出的异常,如抛出整数类型的异常,可以使用throw(int)。

//四、->返回值类型,标识函数返回值的类型,当返回值为void,或者函数体中只有一处return的地方(此时编译器可以自动推断出返回值类型)时,这部分可以省略。

//五、{函数体 },标识函数的实现,这部分不能省略,但函数体可以为空。

案例1:

#include<functional>

#include<iostream>

#include<vector>

#include<algorithm>

usingnamespacestd;

//lambda表达式简单案例

voidmain()

{

//[函数对象参数]、(操作符重载函数参数)、mutable或exception声明、->返回值类型、{函数体 }

autofun1 = [](){cout <<"hellochina"<<endl; };

fun1();

autofun2 = [](inta,intb){returna +b; };

cout <<fun2(10, 9) <<endl;

std::cin.get();

}

运行结果:

案例2:

#include<functional>

#include<iostream>

#include<vector>

#include<algorithm>

usingnamespacestd;

voidmain()

{

vector<int> myv;

myv.push_back(1);

myv.push_back(2);

myv.push_back(11);

autofun1 = [](intv){ cout <<v <<endl; };

//通过这种方式操作myv中的值

for_each(myv.begin(),myv.end(),fun1);

cout <<"----------------" <<endl;

//直接写在内部,下面的v表示传递进lambda表达式的参数

for_each(myv.begin(),myv.end(),[](intv)

{

cout <<v <<endl;

});

std::cin.get();

}

运行结果:

案例3:

#include<functional>

#include<iostream>

#include<vector>

#include<algorithm>

usingnamespacestd;

voidmain()

{

vector<int> myv;

myv.push_back(1);

myv.push_back(2);

myv.push_back(11);

inta = 10;

// =知道a的存在,可以引用,只能读,不可以写,引用当前块语句内部的局部变量

autofun1 = [=](intv){v +=a; cout <<v <<endl; };

for_each(myv.begin(),myv.end(),fun1);

//此时a没有被修改

cout <<a <<endl;

std::cin.get();

}

运行结果:

案例4:

#include<functional>

#include<iostream>

#include<vector>

#include<algorithm>

usingnamespacestd;

voidmain()

{

vector<int> myv;

myv.push_back(1);

myv.push_back(2);

myv.push_back(11);

inta = 10;

//引用变量a,相当于直接操作a

autofun1 = [&a](intv){a = 3; v +=a; cout <<v <<endl; };

for_each(myv.begin(),myv.end(),fun1);

//此时a发生变化

cout <<a <<endl;

std::cin.get();

}

运行结果:

案例5:

#include<functional>

#include<iostream>

#include<vector>

#include<algorithm>

usingnamespacestd;

voidmain()

{

[](){cout <<"hellochina"; };//是一个函数指针

[](){cout <<"hellochina";}();//如果没有定义名称,如果想调用lambda表达式,可以直接在lambda表达式的最后面加上()

cin.get();

}

运行结果:

案例6:

#include<functional>

#include<iostream>

#include<vector>

#include<algorithm>

usingnamespacestd;

classtest

{

public:

vector<int> myv;

intnum;

public:

voidadd()

{

num = 12;

myv.push_back(10);

myv.push_back(11);

//[]引用this

int x = 3;

autofun1 = [this,x](intv){cout <<v+x+this->num << endl; };

//=按照副本引用this,还有当前块语句局部变量,不可以赋值,但是可以读取

//&按照引用的方式操作局部变量,this,可以赋值,可以读取

//副本引用a,[=]  [a]

//引用a [&] [&a]

autofun2 = [&](intv){cout <<v +x +this->num << endl;x = 3; };

for_each(this->myv.begin(),this->myv.end(),fun1);

cout <<"-----------------" <<endl;

for_each(this->myv.begin(),this->myv.end(),fun2);

}

};

voidmain()

{

testt;

t.add();

cin.get();

}

运行结果:

案:7:

#include<functional>

#include<iostream>

#include<vector>

#include<algorithm>

usingnamespacestd;

//返回值案例

voidmain()

{

//double是返回值类型

autofun1 = []()->double{cout << "hello china" <<endl;return 1; };

fun1();

//通过decltype(a/b)的方式获得类型

autofun2 = [](inta,doubleb)->decltype(a / b){cout << "hello china" <<endl;returna /b; };

fun2(1, 2.3);

std::cin.get();

}

运行结果:

案例8:

#include<functional>

#include<iostream>

#include<vector>

#include<algorithm>

usingnamespacestd;

void main()

{

inta = 10;

//mutable使可以改副本了。如果下面的去掉将会报错

autofun1 = [a](intv)mutable->double{v += a; cout <<v <<endl;a = 3;return 3; };

cout <<a <<endl;

std::cin.get();

//运行结果还是10

}

Lambada表达式补充

{

auto func = []{return 1; };

int i = func();

CCLog("i = %d", i);

}

//最简单的lambada表达式是只要一个中括号和一个大括号

//[]捕获列表

//{}函数体

//1.捕获列表,可以放变量名,这里可以用来传递函数体内定义的变量

{

int v = 100;

auto func = [v]{return v; };

int x = func();

}

//2.捕获列表,可以捕获多个变量

{

int p = 100, q = 200;

auto func = [p, q]{return p + q; };

int s = func();

}

// 3.捕获列表,有引用和传值两种方式,传值不可以改变,引用可以改变,并且改变外部的变量值

{

int p = 100, q = 200;

auto func = [p, &q]{q++;  return p + q; };

int s = func();

}

//4.捕获列表,可以定义mutable类型的lambada,能改变传值的捕获参数,

//但是不能改变外部的变量值

{

int p = 100, q = 200;

auto func = [p, q]()mutable{p++; q++; return p + q; };

int s = func();

CCLog("p = %d,q = %d,s = %d", p, q, s);

}

//5.捕获列表,可以用=或者&捕获所有变量,=指传值,&表示引用

{

int p = 100, q = 200;

//用&的时候,所有的都可以调用了,[&,p]:表示除了p不能被使用,其它的都可以被使用

auto func = [&]{

return p + q;

};

}

//稍微复杂点的lambda表达式

{

auto add = [](int v1, int v2){return v1 + v2; };

auto a = add(1 , 2);

}

//小括号中的是参数列表,参数列表和捕获列表区别在于,参数列表的参数由调用方决定,

//捕获列表由定义方决定,所以更加灵活

//更加复杂的lambada表达是,有返回值,返回值一般都省略

{

//->int表示返回值是int类型的

auto add = [](int v1, int v2)->int{return v1 + v2; };

}

//总结:auto func = [](){}

{

auto func = [](){};

}

lambada表达式相关推荐

  1. 2.cocos2dx 3.2中语法的不同之处,lambada表达式的使用和function和bind函数的使用

    1        打开建好的T32  Cocos2dx-3.2的一个项目 2        设置Cocos显示窗口的位置是在AppDelegate.cpp中: 3  设置自适应窗口大小的代码是在上面的 ...

  2. Lambada表达式常用案例(技能题)

    一:Lambada表达式的结构 1)Lambada表达式的结构我们可以看成三部分 () -> {},箭头左边是参数,右边是执行体,箭头是格式 二:使用案例(List为主的操作) 1)实体对象 @ ...

  3. lambada表达式介绍

    前言 在其他语言中,函数式编程是很常见的,java8中为了支持函数式编程,新增了lambada表达式.lambada表达式其实是匿名内部类和闭包的一种符号表示,至于概念性的问题,大家不必深究,lamb ...

  4. JDK1.8之Lambada表达式一

    一. lambada表达式简介 我们知道对于Java变量可以赋给其一个值,而如果想将"一块代码(一个完整的方法)"赋给一个Java变量,如下所示,怎么做呢? 你可能认为就是下面的方 ...

  5. lambada表达式总结

    前言 作为jdk1.8的新特性,8的标准库都开始大量使用lambda表达式了,你有什么理由不去学习lambda,这么简洁,这么爽的一种编程方法,不学不觉得可惜吗? lambda即λ,是匿名函数的意思, ...

  6. Python(十)lambada表达式

    一.Lambda 表达式 概念:是一个匿名函数,Lambda表达式基于数学中的λ演算得名,直接对应于其中的lambda抽象(lambda abstraction),是一个匿名函数,即没有函数名的函数. ...

  7. 第十节 Java工具包-Collections 流--lambada表达式

     collection为核心的接口,包含了多种不同类型的集合以及多种不同数据结构的实现的类 list  set queue都是在collection接口下的所以都有.add()方法 package c ...

  8. Lambada表达式的用法

    Lambada表达式是JDK1.8中最重要的新功能之一.使用Lambada表达式可以替代只有一个抽象函数的接口实现,告别匿名内部类.代码看起来更简洁易懂.Lambada表达式同时还提升了对集合.框架的 ...

  9. lambada表达式理解

    lambada表达式是替换匿名内部类的函数式接口实现类 函数式接口 只有一个抽象方法的接口(可以有默认方法和静态方法),通常带有@FunctionalInterface注解 常用函数式接口 Predi ...

最新文章

  1. 2、MySQL主键(PRIMARY KEY)
  2. Python小知识 | 这些技能你不会?(三)
  3. android WIFI信息获取
  4. operator.ne_Python operator.ne()函数与示例
  5. elementui的css文件没有引入_Python中引入模块详细介绍,使用模块的过程中注意事项教程...
  6. tar 解压到指定目录、去除前导目录
  7. dhcp 服务器发出了 dhcpnack 消息,跪求DHCP报文分析
  8. 《第一本Docker书(修订版)》——第1章_简介_1.1Docker简介
  9. 新浪微博OAuth2.0 VS OAuth1.0 主要区别总结
  10. Jmeter之简单控制器
  11. VirtualBox安装增强功能时报错:未能加载虚拟光盘VBoxGuestAdditions.iso 到虚拟电脑
  12. pyodbc-操作SQLserver
  13. Linux运维学习路线
  14. 让Mac文本编辑器成为HTML编辑器
  15. MS sqlserver数据库恢复出错 Exclusive access could not be obtained because the database is in use
  16. 人机智能的逻辑哲学论
  17. 怪物的生成 攻击和掉落金币
  18. BMap添加海量点数据,BMap.Point携带数据
  19. Windows系统Git安装教程,超详细的安装过程!附软件资料~
  20. 机器视觉-相机选择方法-缺陷检测

热门文章

  1. JavaScript实现计算π值算法(附完整源码)
  2. wxWidgets:wxDialUpManager类用法
  3. wxWidgets:wxDateSpan类用法
  4. boost::test::string_token_iterator相关的测试程序
  5. boost::mp11::mp_iterate相关用法的测试程序
  6. boost::hana::fix用法的测试程序
  7. boost::geometry模块实现自定义Linestring示例
  8. boost::fusion::fused_function_object用法的测试程序
  9. boost::contract模块实现access的测试程序
  10. Boost:以协程的方式实现带有默认值的echo服务器的实例