简介

lambda表达式是一个匿名内联函数,可以定义别的函数体中,其形式如下:

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

从中可以看到,lambda只能使用尾置返回类型

尾置返回类型放在参数列表后边以->开头,然后再原来放返回类型的地方放一个auto。

结构

lambda不可缺少的部分是捕获列表capture list和函数体funtion body,其他都可以忽略不写。

return type

return type会根据funtion body中的内容来推断,如果只含有一个return 语句,类型就从其推断,否则就是void。如果不是上述情况,就需要自己定义一个返回类型,防止推断为void和期望返回的类型不符合的问题。

parameter list

parameter list需要实参和形参互相匹配,且lambda不能有默认参数,所以两者属于也要相等。

capture list

capture list是lambda所在函数中定义的局部变量,lambda可以对其进行捕获然后进行使用,捕获过程也分为值和引用,使用时看是否需要对捕获的变量进行修改和所捕获的变量是否支持这种操作。

值捕获和引用捕获

值捕获需要变量能被拷贝,比如IO流是不能被拷贝,这就只能用引用。

有时可能需要返回一个lambda,当返回时,此lambda就不能用到引用,否则就会出现函数将返回一个局部变量的引用,这是不合法的。

引用捕获需要对象存在。

变量修改

lambda在捕获的时候就对变量进行了复制(值捕获时),所以希望能借此修改外部变量,应该时用引用。但是如果只需要就地对捕获的变量进行修改并且不想对外部修改的话,就可以在值捕获时在参数列表之后指明mutable关键字。

隐式和显式捕获

显示就是直接写出局部变量,隐式就是不在capture list中写出名字,只写捕获方式,其中 = (代表值捕获), & (代表引用捕获),让编译器自己推断要使用那些变量。

比如:

[ = ](string s1){return s1.size()>=length;}  //假设length为一个局部变量,则此时采用值捕获。

两者也可以混搭,这有一个要求:capture list中第一个元素必须为隐式捕获,并且显示和隐式捕获的方式必须不同。

比如:

[ = ,&os](string &s){ os << s << tmp; } //假设tmp为局部变量,采用了值捕获,则只能用显示捕获

捕获列表小结:

列表 含义
[] 空捕获列表
[n1,&n2,n3,&n4...] 以逗号为分隔的捕获列表,显示捕获
[&] 隐式捕获,lambda中使用的变量全部采用引用捕获
[=] 隐式捕获,lambda中使用的变量全部采用值捕获
[&,n1,n2] 混和捕获,n1和n2值捕获,其余为引用捕获
[=,&n1,&n2] 混和捕获,n1和n2引用捕获,其余为值捕获

examples

#include<bits/stdc++.h>
using namespace std;
void elimDups(vector<string>& words) {
sort(words.begin(), words.end());
auto end_unique = unique(words.begin(), words.end());
//for (auto i : words)cout << i << " "; cout << endl;
words.erase(end_unique, words.end());
//for (auto i : words)cout << i << " "; cout << endl;
} //countwords返回一个计数值,表示有多少个元素长度大于等于sz
int countwords(vector<string>& words, vector<string>::size_type sz) {
elimDups(words);//字典序排序,删除重复元素;
stable_sort(words.begin(), words.end(), [](const string& s1, const string& s2) {return s1.size() < s2.size(); });//长度排序,长度相同也维持字典序
//for (auto i : words)cout << i << " "; cout << endl;
auto it = count_if(words.begin(), words.end(), [sz](const string& s) { return s.size() >= sz; });
return it;
}
int main(void) {
vector<string>words;
string str;
while (cin >> str) {
words.push_back(str);
}
//for (auto i : words)cout << i << " "; cout << endl;
int cnt = countwords(words, 5);//计算有多少个元素长度大于等于5
cout << cnt << endl;
//the quick red fox jumps over the slow red turtle
auto f=[&]()->bool {
while (cnt > 0)cnt--;
if (cnt == 0)return true;
};
cout << f() << endl;//调用lambda改变变量
cout << cnt << endl;
cout << f() << endl;//再调用lambda不再改变变量
cout << cnt << endl;
return 0;
}
input:
the quick red fox jumps over the slow red turtle
output:
3
1
0
1
0

problems

1.为什么需要使用mutable才能进行修改?

lambda在定义时,编译器就会生成一个对应的匿名类,该类会包含所有被捕获的变量作为数据成员初始化,并且该类的operator()默认时const,这就导致无法修改类中的成员函数,加上mutable,即可转换为非const,就可以就地修改了。

C++11 lambda表达式小结的更多相关文章

  1. C++11 lambda 表达式

    C++11 新增了很多特性,lambda 表达式是其中之一,如果你想了解的 C++11 完整特性,建议去这里,这里,这里,还有这里看看.本文作为 5 月的最后一篇博客,将介绍 C++11 的 lamb ...

  2. C++11 lambda 表达式解析

    C++11 新增了很多特性,lambda 表达式是其中之一,如果你想了解的 C++11 完整特性,建议去这里,这里,这里,还有这里看看.本文作为 5 月的最后一篇博客,将介绍 C++11 的 lamb ...

  3. 详解 C++11 lambda表达式

    详解 C++11 lambda表达式   lambda表达式是函数式编程的基础.咱对于函数式编程也没有足够的理解,因此这里不敢胡言乱语,有兴趣的可以自己查找相关资料看下.这里只是介绍C++11中的la ...

  4. C++11 lambda表达式学习

    lambda表达式是函数式编程的基础.咱对于函数式编程也没有足够的理解,因此这里不敢胡言乱语,有兴趣的可以自己查找相关资料看下.这里只是介绍C++11中的lambda表达式自己的认识.这里有参考文档h ...

  5. C++ 11 Lambda表达式

    C++11的一大亮点就是引入了Lambda表达式.利用Lambda表达式,可以方便的定义和创建匿名函数.对于C++这门语言来说来说,“Lambda表达式”或“匿名函数”这些概念听起来好像很深奥,但很多 ...

  6. C++11 lambda表达式是如何实现的?

    lambda表达式是如何实现的呢? 其实是编译器为我们了创建了一个类,这个类重载了(),让我们可以像调用函数一样使用.所以,你写的lambda表达式和真正的实现,是这个样子的: 而对于捕获变量的lam ...

  7. C++ 11 Lambda表达式、auto、function、bind、final、override

    接触了cocos2dx 3.0,就必须得看C++ 11了.有分享过帖子:[转帖]漫话C++0x(四) —- function, bind和lambda.其实最后的Lambda没太怎么看懂. 看不懂没关 ...

  8. c++11——lambda表达式

    lambda表达式 函数式编程的一个语法,有如下优点: (1)声明式编程风格:就地匿名定义目标函数或函数对象,不需要额外写一个命名函数或者函数对象.以更直接的方式写程序,好的可读性和可维护性. (2) ...

  9. C++11 Lambda表达式简单解析

    C++11 新增了非常多特性,lambda 表达式是当中之中的一个.假设你想了解的 C++11 完整特性, 建议去http://www.open-std.org/看看新标准! 非常多语言都提供了 la ...

随机推荐

  1. kettle入门大数据管理工具

    研究 kettle 的使用 大佬博客:https://www.cnblogs.com/mq0036/p/9238646.html 国内镜像下载:http://mirror.bit.edu.cn/pen ...

  2. 红帽学习笔记[RHCSA] 第二课[文件、目录、相关命令]

    第二课 常用的目录结构与用途 / 根目录 /boot 存储的是系统起动时的信息和内核等 /dev 存储的是设备文件 /etc 存储的是系统的配置文件 /root 存储的是root用户的家目录 /hom ...

  3. 如何在centos7中显示/etc/目录下以非字母开头,后面跟了一个字母及其它任意字符的文件或目录

    ls /etc |grep "^[^[:alpha:]][[:alpha:]].*"

  4. kubeadm初始化kubernetes集群

    有两种方式安装集群: 1.手动安装各个节点的各个组件,安装极其复杂困难. 2.使用工具:kubeadm kubeadm 是官方提供的专门部署集群的管理工具. 1. 在kubeadm下每个节点都需要安装 ...

  5. Spring Boot 2.2.0 正式发布,支持 JDK 13!

    Java技术栈 www.javastack.cn 优秀的Java技术公众号 推荐阅读: Spring Boot 2.2.0 正式发布了,可从 repo.spring.io 或是 Maven Centr ...

  6. Tomcat 一台机器运行多个Tomcat

    转 https://www.cnblogs.com/andy1234/p/8866588.html 在一台Win10 PC 上面同时开启两个Tomcat系统为例. 1. 硬件环境 2. 到Tomcat ...

  7. GmSSL Build with VS2017

    使用背景: 最近研究GB35114, 有关于sip协议部分,exosip的已经编译过,由于gb3511中采用的是国密算法,因此这里记录一下GMSSL在windows下的编译过程以及遇到的错误 详细GM ...

  8. Python2 中 range 和 xrange 的区别?

    两者用法相同,不同的是 range 返回的结果是一个列表,而 xrange 的结果是一个生成器,前者是直接开辟一块内存空间来保存列表,后者是边循环边使用,只有使用时才会开辟内存空间,所以当列表很长时, ...

  9. Springboot+CAS单点登录

    一:安装CAS 下载cas:https://github.com/apereo/cas 1.1 将cas并打成war包.放入一个干净的tomcat中,启动tomcat测试: http://localh ...

  10. go & RabbitMQ

    参考 RabbitMQ tutorial - 官方示例 Go code for RabbitMQ tutorials - 官方示例源码 go语言开发RabbitMQ-牛刀小小试试