C++ Primer 笔记——lambda表达式
1.一个lambda表达式表示一个可调用的代码单元,可以理解为一个未命名的内联函数,但是与函数不同,lambda表达式可能定义在函数内部。其形式如下:
[capture list] (parameter list) -> return type { function body }
- capture list 是一个lambda所在函数中定义的局部变量的列表(通常为空)
- return type, parameter list 和 function body与任何普通函数一样,分别表示返回类型,参数列表和函数体
- lambda必须使用尾置返回
- 我们可以忽略参数列表和返回类型,但必须永远包含捕获列表和函数体
- 如果忽略返回类型,lambda会从函数体推断出返回类型。
- 如果lambda的函数体包含任何单一return语句之外的内容,且未指定返回类型,则返回void
- 捕获列表只用于局部非static变量,lambda可以直接使用局部static变量和在它所在函数之外声明的名字。
void test()
{
int i = ;
int j = ;
auto f = [i, j](int base) -> int { return base + i + j; }; // i,j必须在捕获列表里面,这里的返回类型int其实可以省略
int k = f(); // k的结果为3
}
2.当定义一个lambda时,编译器生成一个与lambda对应的新的(未命名的)类类型。默认情况下这个类都包含一个对于该lambda所捕获的变量的数据成员。类似任何普通类的数据成员,lambda的数据成员也在lambda对象创建时被初始化。
3.与传值参数类似,采用值捕获的前提是变量可以拷贝,与参数不同,被捕获的变量的值是在lambda创建时拷贝,而不是调用时拷贝。
void test()
{
int i = ;
auto f = [i] {return i + ; }; // 此时i的值就已经被拷贝了
int j = f();
}
4.当以引用方式捕获一个变量时,必须保证在lambda执行时变量是存在的。
void test()
{
int i = ;
auto f = [&i] {return ++i; };
int j = f(); // 此时i等于11
}
5.在捕获列表中写一个 &或= 可以告诉编译器我们想采用值捕获还是引用捕获,这种方法叫做隐式捕获。
void test()
{
int i = ;
auto f = [&] {return ++i; }; // 采用引用捕获
int j = f();
}
6.我们也可以混合使用值捕获或者引用捕获。当使用混合方式的时候,显示捕获的方式不可以与隐式捕获的方式相同。
void test()
{
int i = ;
int j = ;
auto f = [=, &i] {i++; return i + j; }; // i采用引用捕获,其他采用值捕获
int k = f(); // k等于20
}

7.默认情况下,对于一个值被拷贝的变量,lambda不会改变其值,如果我们希望能改变一个被捕获的变量的值,就必须在参数列表首加上mutable。
void test()
{
int i = ;
auto f = [i] (){return ++i; }; // 错误,不可以改变i的值
auto f1 = [i] () mutable {return ++i; }; // 正确
}
8.bind函数(头文件 functional中)可以看作一个通用的函数适配器,它接受一个可调用对象,生成一个新的可调用对象来适应原对象的参数列表。调用bind的一般形式为:
auto newCallable = bind(callable, arg);
void test(int i, int j)
{
std::cout << i - j << std::endl;
} auto test0 = std::bind(test, , );
test0(); // 结果4 auto test1 = std::bind(test, std::placeholders::_1, , std::placeholders::_2, ); // 错误,不可以这么写
test1(); auto test2 = std::bind(test, std::placeholders::_1, );
test2(); // 结果-4 auto test3 = std::bind(test, std::placeholders::_1, std::placeholders::_2);
test3(, ); // 结果1 auto test4 = std::bind(test, std::placeholders::_2, std::placeholders::_1); // 参数顺序可以重排
test4(, ); // 结果-1 auto test5 = std::bind(test, , std::placeholders::_1); // 注意这个占位符占的是test5()的参数位置,而不是test()的
test5(); // 结果-4
以上代码请注意命名空间,否则会和socket的bind函数产生二义性。
9.如果我们想绑定的参数无法拷贝,或者我们想引用时应该使用ref函数。
void test(int& i, int j)
{
std::cout << i - j << std::endl;
} int i = ;
auto test0 = std::bind(test, std::ref(i), );
test0(); // 结果4
10.编译器将lambda表达式翻译成一个未命名类的未命名对象,在lambda表达式产生的类中含有一个重载的函数调用运算符。这个类不含默认构造函数,赋值运算符及默认析构函数。它是否有默认的拷贝/移动构造函数则通常要视捕获的数据成员类型而定。
std::size_t i;
auto test_func = [i](const std::string& str) {return str.size() > i; }; class test
{
public:
test(std::size_t size) : m_size(size) {}
bool operator()(const std::string& str) { return str.size() > m_size; }; private:
std::size_t m_size;
};
C++ Primer 笔记——lambda表达式的更多相关文章
- Java8学习笔记----Lambda表达式 (转)
Java8学习笔记----Lambda表达式 天锦 2014-03-24 16:43:30 发表于:ATA之家 本文主要记录自己学习Java8的历程,方便大家一起探讨和自己的备忘.因为本人 ...
- .net学习笔记---lambda表达式(自执行方法)
http://www.cnblogs.com/jesse2013/p/happylambda.html#b034 lambda表达式 http://www.cnblogs.com/OceanEyes/ ...
- Java学习笔记-Lambda表达式
Lambda表达式支持将代码块作为方法参数,Lambda表达式允许使用简洁的代码来创建只有一个抽象方法的接口(这种接口被称为函数是接口)的实例 意义 自从Java 8开始,Java支持Lambda表达 ...
- Java疯狂讲义笔记——Lambda表达式
Java8新增的Lambda表达式 [特性]支持将代码块作为方法参数,Lambda表达式允许使用更简洁的代码来创建只有一个抽象方法的接口(这种接口被称为函数式接口)的实例. [组成部分]1,形参列表 ...
- 0028 Java学习笔记-面向对象-Lambda表达式
匿名内部类与Lambda表达式示例 下面代码来源于:0027 Java学习笔记-面向对象-(非静态.静态.局部.匿名)内部类 package testpack; public class Test1{ ...
- 委托学习笔记后续:泛型委托及委托中所涉及到匿名方法、Lambda表达式
引言: 最初学习c#时,感觉委托.事件这块很难,其中在学习的过程中还写了一篇学习笔记:委托.事件学习笔记.今天重新温故委托.事件,并且把最近学习到和委托相关的匿名方法.Lambda表达式及泛型委托记录 ...
- 《C#本质论》读书笔记(12)委托和Lambda表达式
12.1.委托概述 12.1.2 委托的数据类型 为了减少重复代码数量,可以将比较方法作为参数传递给 BubbleSort()方法.此外,为了将方法作为参数传递,必须有一个能够标识方法的数据类型--也 ...
- Java开发笔记(六十一)Lambda表达式
前面介绍了匿名内部类的简单用法,通过在sort方法中运用匿名内部类,不但能够简化代码数量,还能保持业务代码的连续性.只是匿名内部类的结构仍显啰嗦,虽然它省去了内部类的名称,但是花括号里面的方法定义代码 ...
- lambda表达式笔记
前几天一位好友分享了一篇文章,其中讲到了lambda表达式,正好最近看了一些内容,就做做笔记吧... lambda表达式服务于函数式接口,如果需要一个函数式接口的对象时,就可以用lambda表达式代替 ...
随机推荐
- 顶层const
顶层const:指针本身是一个常量 底层const:指针所指对象是一个常量 顶层const可以表示任意的对象是常量,这一点对任何数据类型都适用,如算术类型.类.指针等 底层const则与指针和引用等复 ...
- ARM核心板_迅为imx6工控核心板_核心板中的小新潮
ARM核心板_迅为imx6工控核心板_核心板中的小新潮核心板参数 尺寸 51mm*61mm CPU Freescale Cortex-A9 四核 i.MX6Q,主频 1.2 GHz 内存 2GB DD ...
- Css - 浮动布局
Css - 浮动布局 浮动布局 float 取值:left | right | none 利用float属性可设置元素的浮动,虽然浮动主要是应用于块元素,但行内元素其实也可以浮动,但行内元素本来就是一 ...
- MySQL绿色版安装整套流程
https://www.cnblogs.com/LiuChunfu/p/6426918.html
- 什么是联机分析处理(OLAP)
OLAP的提出引起了很大的反响,OLAP作为一类产品同联机事务处理 (OLTP) 明显区分开来. 当今的数据处理大致可以分成两大类:联机事务处理OLTP(on-line transaction pro ...
- QR 编码原理(三)
一.日本汉字(KANJI)是两个字节表示的字符码,编码的方式是将其转换为13字节的二进制码制. 转换步骤为: 1.对于JIS值为8140(hex) 到9FFC(hex)之间字符: a)将待转换的JIS ...
- LwIP Application Developers Manual6---Application API layers
1.前言 lwIP提供3种应用编程接口来跟TCP/IP内核通信,如下所示: 低水平的内核/回调或raw API 2个高水平序列API: 1) netconn API 2) socket API(为了兼 ...
- python使用pudb调试
pudb是pdb的升级版本 安装 pip3 install pudb 使用方法 在程序文件的开头导入包 from pudb import set_trace set_trace()#断点位置 运行的时 ...
- lnmp 搭建 初试
#初始化环境检查 # uname -r -.el6.x86_64 # uname -m x86_64 #添加mysql用户 useradd -s /sbin/nologin mysql -M #下载安 ...
- boost.asio包装类st_asio_wrapper开发教程(一)
一:什么是st_asio_wrapper它是一个c/s网络编程框架,基于对boost.asio的包装(最低在boost-1.49.0上调试过),目的是快速的构建一个c/s系统: 二:st_asio_w ...