C++11之lambda表达式解析
什么是Lanmbda?
简短函数,就地书写。常用于向函数(算法)传递函数参数。
语法
Lambda 表达式,[capture](paras)mutable->return type{statement} 全部语法格式,分如下章节介绍:
- [capture]: 捕获列表。捕获列表,总是出现在 lambda 函数的开始处。事实上[ ]是lambda 的引用符。换句话说,编译器根据引出符判断接下来的代码是否是 lamba 函数。
- (paramers): 参数列表。与普能函数的参数列表一致。如果不需要传递参数,可以连同()一起省略。
- mutable: 默认情况下,lambda 函数总是一个 const 函数,mutable 可以取消其常量性。在使用该修饰符时,参数列表不可以省略(即使参数为空)。
- ->return-type: 返回类型。用于追踪返回类型形式声明函数的返回类型。出于方便,不需要返回值的时候可以连同->一起省略。此外返回类型明确的情况下,也可以省略该部分。编译器可以自行推导。
- {statement}: 函数体。内容与普通函数一样,不过除了可以使用参数之外,还可以使用所有捕获的变量。
格式1——[]{}闭包+函数体
#include <iostream>
using namespace std; int main(int argc, char *argv[])
{
auto foo = []{ return + ;};
cout<<foo();
cout<<[]{ return + ;}()<<endl;
return ;
}
格式2——[](){}闭包+参数+函数体
格式3——[]()->{}闭包+参数+返回值+函数体
#include <iostream>
using namespace std;
int main(int argc, char *argv[])
{
auto foo = [](int x, int y)->int{return x + y;};
cout<<foo(,)<<endl;
cout<<[](int x, int y){return x + y;}(,)<<endl;
return ;
}
格式4——[]()mutable->{}闭包+参数+可修改+返回值+函数体
#include <iostream>
using namespace std;
int main(int argc, char *argv[])
{
int x = ; int y = ;
cout<<"main:"<<x<<y<<endl;
auto foo = [=]()mutable{
x = ;
y = ;
cout<<"lambda:"<<x<<y<<endl;
};
foo();
cout<<"main:"<<x<<y<<endl;
return ;
}
———————————————模块说明—————————————————
[]闭包
lambda 函数能够捕获 lambda 函数外的具有自动存储时期的变量。函数体与这些变
量的集合合起来叫闭包。闭包的概念在 lambda 中通过[]来体现出来。
- [] 不截取任何变量。[bar] 仅对外部变量 bar 值传递在函数体中使用。
- [&bar] 仅对外部变量 bar 引用传递在函数体中使用。
- [x, &y] 仅 x 按值传递,y 按引用传递在函数体中使用。。
- [&} 截取外部作用域中所有变量,并作为引用传递在函数体中使用。
- [=] 截取外部作用域中所有变量,并按值传递在函数体中使用。
- [=, &foo] 截取外部作用域中所有变量,并值传递在函数体中使用,但是对 foo变量使用引用传递。
- [&, =foo] 截取外部作用域中所有变量,在函数体中作引用传递使用,但是对foo 变量作值传递。
上述,中涉及到值传递要发生拷贝行为,而引用传递则不会发生拷贝行为。捕获列表中不允许重复。比如:[=, a] [&,&this]。闭包的本质,初始化 lamda 表达式。
mutable 作用——截取值 还是截取引用?
int main()
{
int i=;
auto f=[i](){return i+=;};
i=;
auto j=f();
cout<<"i="<<i<<endl;
cout<<"j="<<j<<endl; return ;
}
运行此函数会提示:error: assignment of read-only variable 'i' auto f=[i](){return i+=5;};
说明{return i+=5;}内的i是受到保护的,无法修改。如果此时将[i](){return i+=5;};改为 [&i](){return i+=5;};就能通过编译,由于传入的是引用,所以i的改变是
执行结果:
i=5;
j=5;
mutable加持!
int main()
{
int i=;
auto f=[i](){return i+=;};
i=;
auto j=f();
cout<<"i="<<i<<endl;
cout<<"j="<<j<<endl;
auto f1=[i](){return i+=;};
cout<<"k="<<k<<endl;
return ;
}
如果将表达式改为[i]()mutable{return i+=5;};也可以通过,mutable改变了{return i+=5;} i 的const属性,也可以通过编译。
运行结果:
i=0;
j=47;
K=5;
这说明了什么?
lambda表达式中[i]值的捕获是发生在编译期的!执行时此时 i=42 值早已拷贝传入。所以 i 之后的改变对于表达式运行并无任何影响。
也就是说,当声明时lambda表达式对外部变量(i)以赋值的方式捕获后赋给一个“函数指针”,执行该函数指针时,捕获的变量 (i) 不与外部发生任何关系,无论外部变量如何改变。要想重新“刷新”内部变量,需要重新声明该lambda表达式。
mutable参数只对只[]闭包中捕获的(非引用)值才有作用。对于()传入的参数没有任何影响。
C++11之lambda表达式解析的更多相关文章
- C++11 lambda 表达式解析
C++11 新增了很多特性,lambda 表达式是其中之一,如果你想了解的 C++11 完整特性,建议去这里,这里,这里,还有这里看看.本文作为 5 月的最后一篇博客,将介绍 C++11 的 lamb ...
- 「C++11」Lambda 表达式
维基百科上面对于 lambda 的引入是如下描述的: 在标准 C++,特别是当使用 C++ 标准程序库算法函数诸如 sort 和 find.用户经常希望能够在算法函数调用的附近定义一个临时的述部函数( ...
- C++11之lambda表达式
lambda表达式源于函数式编程的概念,它可以就地匿名定义目标函数或函数对象,不需要额外写一个命名函数或者函数对象.lambda表达式的类型在C++11中被称为"闭包类型",也可以 ...
- SqlHelper简单实现(通过Expression和反射)5.Lambda表达式解析类
这个ExpressionHelper类,是整个SqlHelper中,最核心的一个类,主要功能就是将Lambda表达式转换为Sql语句.同时这个转换过程比较复杂,所以下面详细讲解一下思路和这个类的作用. ...
- C++11 里lambda表达式的学习
最近看到很多关于C++11的文档,有些是我不怎么用到,所以就略过去了,但是lambda表达式还是比较常用的,其实最开始学习python的时候就觉得lambda这个比较高级,为什么C++这么弱.果然C+ ...
- 【C++11】 lambda表达式
i.e.int x = 10;int y = 20;int z = [&]{ x = x * x; y = y * y; return x + y;}(); 上面z后面以[]开头的为一个lam ...
- C++11之lambda表达式应用
应用 foreach语句中 #include <time.h> #include <algorithm> using namespace std; void func(int ...
- c++11:lambda表达式的使用
lambda表达式的一般形式: [capture list] (parameter list) -> return type{function body}; 其中,capture list (捕 ...
- STL - C++ 11的Lambda表达式(上)
Lambda始自C++ 11,是一种在表达式或语句内指定函数行为的定义式. 你可以定义函数行为作为对象,以inline实参的形式传给算法作为predicate(判断式). eg: std:transf ...
随机推荐
- json 数据返回解密
http://www.cnhan.com/shantui//dynamic/get/data/allCompanyInfoByCompCode.json?compCode=6SU5YCJ <sc ...
- Swift 学习笔记 (枚举)
枚举为一种相关值定义了一个通用类型,从而可以让你在代码中类型安全的操作这些值. Swift中的枚举很灵活,不需要给每一个枚举中的成员都提供值.如果一个值(所谓 原时值) 要被提供给每一个枚举成员,那么 ...
- 如何在ubuntun中安装pycharm并将图标显示在桌面上
安装pycharm首先要安装jdk. 可以通过java -V来查看是否安装了jdk.安装jdk的方法如下: 1 首先在oracle网站下载jdk,现在jdk是1.8的. 2 新建一个/usr/lib/ ...
- BZOJ 2142 礼物 数论
这道题是求组合数终极版. C(n,m) mod P n>=1e9 m>=1e9 P>=1e9且为合数且piqi<=1e5 拓展lucas定理. 实际上就是一点数论小知识的应用. ...
- vue 升降排序
本实例是根据工作进度的百分比来进行排序. html <div class="ibox-content"> <li v-for="(rangeItem,i ...
- java对象的初始化过程和创建对象的几种方式
1.加载父类,加载父类的静态属性和静态代码块 2.加载子类,加载子类的静态属性和静态代码块 3.初始化父类中的非静态属性并赋初值,执行父类非静态代码块,执行父类构造. 4.初始化子类中的非静态属性并赋 ...
- Go丨语言package github.com/Go-SQL-Driver/MySQL: exec: "git": executable file not found in %PATH%解决方法
Go语言在添加第三方MySQL驱动的时候报错: go: missing Git command. See https://golang.org/s/gogetcmd package github.co ...
- L97
We are young. So let's set the world on fire. We can burn brighter than the sun.我们是青年,让我们点亮世界,释放生命,胜 ...
- 前端多媒体(4)—— video标签全面分析
测试地址:https://young-cowboy.github.io/gallery/html5_video/index.html 属性 一些属性是只读的,一些属性是可以修改从而影响视频播放的. a ...
- intent实现Activity之间跳转的各种传值
一.在Activity之间传递String类型的数据 传递 @Override public void onClick(View v) { String num1 = firstNum.getText ...