一、lambda表达式

lambda表达式原型: [capture list] (parameter list) -> retrue type { function body }
一个lambda表达式表示一个可调用的代码单元,可以理解为一个未命名的内联函数。一个lambad表达式包含 : 一个捕获列表,一个参数列表,一个返回类型和函数体。
但与函数不同的是,lambda可能定义在函数内部。 lambda可以忽略参数列表和返回类型,但必须包含捕获列表和函数体。 
如果一个lambda的函数体包含任何单一的return语句之外的内容,且未指定返回类型,则返回void
1. 捕获列表
lambda的捕获列表只用于局部的非static变量,它可以直接使用局部static变量和它所在函数之外声明的名字。lambda只有在其捕获列表中捕获了一个它所在函数中的局部非static变量,它的函数体才可以使用该变量。
捕获方式有两种 : 值捕获和引用捕获 另外还有 隐式捕获
值捕获 : 和传值参数类似,lambda进行值捕获的时候拷贝变量,与参数不同,lambda的值捕获进行拷贝时是创建时拷贝,而不是调用时拷贝。
引用捕获:引用捕获和引用传递参数类似,不拷贝变量,但应该注意的是,我们lambda在进行引用捕获时,要确保被引用的对象在lambda执行的时候是存在的。
 
隐式捕获 : 值捕获和引用捕获都是显示列出了我们想要捕获的变量,我们还可以让编译器根据lambda体中的代码来推断我们要使用哪些变量,我们需要在捕获列表里写一个 & 或者 =。 &告诉编译器采用引用捕获方式,= 是值捕获方式。还可以混合使用显示和隐式捕获方式。
当我们混合使用显示和隐式捕获方式时,捕获列表中的第一个元素必须是一个&或=, 指定默认的捕获方式为值或引用。  而且显示捕获的变量必须使用与隐式捕获不同的凡是。如果隐式捕获时引用方式,则显示捕获必须是值方式。
lambda捕获列表
                 [ ] 空捕获列表,lambda不能使用所在函数中的变量
            [names] names是一个逗号分隔的名字列表,都是lambda所在函数的变量,默认情况下,捕获列表中的变量都被拷贝
                [&] 隐式捕获列表,采用引用捕获方式。lambda体中所使用的来自所在函数的实体都采用引用方式使用
                [=] 隐式捕获列表,采用值捕获方式。lambda体中所使用的来自所在函数的实体都采用饮用方式使用
                                                                                                                             [&,identifer_list]                          identifer_list是一个逗号分隔的名字列表,包含0个或多个来自所在函数的变量,这些变量都采用值捕获方式,而不在这些列表中的变量则使用引用捕获方式, identifer_list中变量前不能使用&
                                             [=,identifer_list]  identifer_list是一个逗号分隔的名字列表,包含0个或多个来自所在函数的变量,这些变量都采用引用捕获方式,而不在这些列表中的变量则使用值捕获方式,identifer_list中的名字不能包括this,且这些名字前必须使用&

默认情况下,值捕获的变量,lambda不会改变其值,如果我们希望改变一个被值捕获的变量,就把关键字 mutable 跟随在参数列表后
void func(){
size_t i = 0;
auto f = [i] () mutable { return ++i; }
cout << "i = " << i << endl;
}

输出 i = 1

2. 指定lambda返回类型
前面我们说,如果一个lambda的函数体包含任何单一的return语句之外的内容,且未指定返回类型,则返回void。
transform(v.begin(), v.end(), v.begin(), [] (int i) {if (i < 0) return -i; else return i;})

这段程序将产生错误,因为编译器推断出lambda返回一个void,但它返回一个int值。

我们需要使用尾置返回类型来指定lambda的返回类型:
transform(v.begin(), v.end(), v.begin(), [] (int i) -> int {if (i < 0) return -i; else return i;})

二、参数绑定 bind函数

标准库bind函数定义在头文件 <functional>中,它接受一个可调用对象,来生成一个新的可调用对象来适用原对象的参数列表。
调用bind的一般形式为:
auto newCallable = bind(callable, arg_list);

newCallable本身是一个可调用对象,arg_list是一个逗号分隔的参数列表,对应给定的callable的参数。当我们调用newCallable时,newCallable会调用callable,并传递给callable对应的参数arg_list。

 arg_list中会出现形如 _n的名字,n是一个整数,从1开始。这些参数是"占位符", 表示newCallable的参数。它们占据了传递给newCallable的参数的位置,数值n表示生成的可调用对象中参数的位置。
名字 _n 都定义在一个名为placeholders的命名空间里,而这个命名空间则包含在std命名空间里。我们要用这些占位符时,就需要using声明:
using std::placeholders::_n

与bind一样,placeholders命名空间也定义在 <functional> 头文件中。
bind的参数 
例:
// g是一个有两个参数的可调用对象
auto g = bind(f, a, b, _2, c, _1);

这样一来,我们传递给g对象的参数,按位置绑定到占位符上,第一个参数绑定到_1, 第二个绑定到_2.

实际上,这个bind调用会将  g(_1, _2) 映射为 f (a, b, _2, c, _1),例如调用g(X, Y)会调用 f (a, b, Y, c, X)
默认情况下,bind的那些不是占位符的参数被拷贝到返回的可调用对象中。对有些绑定的参数我们希望以引用方式传递,或是要绑定参数的类型无法拷贝。
// 错误,不能拷贝os
for_each(words.begin(), words.end(), bind(print, os, _1, ' '));

我们不能拷贝一个ostream对象,如果我们希望传递给bind一个对象但不拷贝它,就必须使用标准库ref函数。

函数ref返回一个对象,包含给定的引用。还有一个cref函数,生成一个const引用的类。ref和cref函数也定义在 <functional>头文件中。

C++ Primer : 第十章 : 泛型算法 之 lambda表达式和bind函数的更多相关文章

  1. C++ Primer : 第十章 : 泛型算法 之 只读、写和排序算法

    大多数算法都定义在<algorithm>头文件里,而标准库还在头文件<numeric>里定义了一组数值泛型算法,比如accumulate. ●  find算法,算法接受一对迭代 ...

  2. lambda表达式与bind函数

    #include<iostream> #include<algorithm> #include<sstream> #include<vector> #i ...

  3. C# 委托 (一)—— 委托、 泛型委托与Lambda表达式

    C# 委托 (一)—— 委托. 泛型委托与Lambda表达式 2018年08月19日 20:46:47 wnvalentin 阅读数 2992   版权声明:此文乃博主之原创.鄙人才疏,望大侠斧正.此 ...

  4. kotlin之lambda表达式和匿名函数

    lambda表达式,称为匿名函数,是一种函数字面值,也就是没有声明的函数,但可以作为表达式传递出去. 函数类型: 对于接受另一个函数的作为自己的参数,必须针对这个参数指定一个函数的类型如 fun &l ...

  5. Python函数与lambda 表达式(匿名函数)

    Python函数 一.函数的作用 函数是组织好的,可重复使用的,用来实现单一或相关联功能的代码段 函数能提高应用的模块性和代码的重复利用率 python 内置函数:https://docs.pytho ...

  6. C# 委托(delegate)、泛型委托和Lambda表达式

    目录 # 什么是委托 # 委托声明.实例化和调用 1.声明 2.委托的实例化 3.委托实例的调用 4.委托完整的简单示例 #泛型委托 1.Func委托 2.Action委托 3.Predicate委托 ...

  7. Lambda表达式公共拼接函数(原创)

    #region Lambda公共拼接函数 /// <summary> /// LambdaWhere(枚举) /// </summary> public enum Lambda ...

  8. 第三天 函数 三元运算 lambda表达式 内置函数 文件操作

    面向过程: 直接一行一行写代码,遇到重复的内容复制黏贴. 不利于代码阅读 代码没有复用 面向对象 将代码块定义为函数,以后直接调用函数 增强了复用性 函数的定义方法 def 函数名(传递参数): 函数 ...

  9. lambda表达式,map函数

    lambda只是一个表达式,不需要定义函数,故也是匿名函数,用法为:lambda 参数:表达式. x=5 list1=[2,3,4] list2=[10,20,30] s=lambda x:x**3 ...

随机推荐

  1. Linux中如何让命令在后台运行

    1.在下达的命令后面加上&,就可以使该命令在后台进行工作,这样做最大的好处就是不怕被ctrl+c这个中断指令所中断. 2. 那大家可能又要问了,在后台执行的程序怎么使它恢复到前台来运行呢?很简 ...

  2. elasticsearch插件之一:marvel

    在 安装插件的过程中,尤其是安装Marvel插件遇到了很多问题,又要下载license.Marvel-agent,又要下载安装Kibana,很多内容 不知道为何这样安装处理.仔细看了看ElasticS ...

  3. C#窗体的加载等待(BackgroundWorker控件)实现

    窗体拉一个Button按钮和一个加载等待显示的label, label默认隐藏,点击按钮时显示这个label,加载完再隐藏 1.工具箱拉BackgroundWorker控件到窗体 2.backgrou ...

  4. 火狐和IE的window.event对象详解(转载)

    FF的FIREBUG,不仅能测试JS还能检查CSS错误,是一般常用的. 但它主要检查FF方面的错误,对IE就无能为力了. 要测试IE,就用ieTester,它可以测试IE几乎所有版本(1.0恐怕也用不 ...

  5. 赋值运算符、拷贝初始化和this指针

    一.赋值运算符和拷贝构造函数(重载技术) 赋值运算符和拷贝构造函数有编译器默认提供,但如果想做更复杂的事,需要重载. 1.下面用一个简单的例子先区分一下赋值运算符和拷贝构造函数: #include&l ...

  6. RPI学习--环境搭建_串口连接

    有两种, 一种是通过MAX2323芯片连接的串口,要接VCC为芯片供电. 另一种是通过PL2302芯片连接的USB,可不接VCC,用电脑USB口为芯片供电. 下面以通过MAX2323方式为例. 1,V ...

  7. windows8.1安装

    不小心下载了英文版的windows8.1的操作系统,要添加中文语言,结果遇到不少问题. 第一:安装中文语言包: 可以在控制面板-添加语言中添加,这个方法好像只能在线更新,那速度,不能忍.还可以下载离线 ...

  8. 使用windows远程桌面连接Windows Azure中的Ubuntu虚拟机

    1.创建ubuntu虚拟机,这里同样不再赘述,创建过程和创建Windows虚拟机基本一样,只是登录可以选择密钥注入或者用户名密码(为了方便我选择了用户名密码认证),创建完成后,查看虚拟机详情中的端口信 ...

  9. (转)面向移动设备的HTML5开发框架

    (原)http://www.cnblogs.com/itech/archive/2013/07/27/3220352.html 面向移动设备的HTML5开发框架   转自:http://blogrea ...

  10. (转) Tomcat部署Web应用方法总结

    原文:http://blog.csdn.net/yangxueyong/article/details/6130065 Tomcat部署Web应用方法总结 分类: Java web2011-01-11 ...