C++11的lambda表达式

        lambda来源于函数式编程的概念。C++11这次终于把lambda加进来了。

        lambda表达式有如下优点:

        1、声明式编程风格:就地匿名定义目标函数或函数对象,不需要额外写一个命名函数或者函数对象。以更直接的方式去写程序,好的可读性和可维护性。

        2、简洁:不需要额外再写一个函数或者函数对象,避免了代码膨胀和功能分散,让开发者更加集中精力在手边的问题,同时也获取了更高的生产率。

        3、在需要的时间和地点实现功能闭包,使程序更灵活。

lambda表达式的概念和基本用法

        lambda表达式定义了一个匿名函数,并且可以捕获一定范围内的变量。lambda表达式的语法形式可简单归纳如下:

[capture](params) opt -> ret { body; };

        其中:capture是捕获列表;params是参数表;opt是函数选项;ret是返回值类型;body是函数体。

        因此,一个完整的lambda表达式看起来像这样:
 

        auto f = [](int) -> int { return a + 1;};cout << f(1) << endl;                //输出:2

        可以看到,上面通过一行代码定义了一个小小闭包,用来输入加1并返回。

        在C++11中,lambda表达式的返回值是通过前面介绍的返回值后置语法来定义的。其实很多时候,lambda表达式的返回值是非常明显的,比如上例。因此,C++11中运行省略lambda表达式的返回值定义:

       

 auto f = [](int a) {return a + 1; };

        这样编译器就会根据return语句自动推导出返回值类型。

        需要注意的是,初始化列表不能用于返回值的自动推导。
 

        auto x1 = [](int i){ return i; };         //OK:return type is intauto x2 = [](){ return {1, 2};};        // error:无法推导出返回值类型

        这时我们需要显示给出具体的返回值类型。

        另外,lambda表达式在没有参数列表时,参数列表是可以省略的。因此像下面的写法都是正确的:
 

        auto f1 = [] () { return 1; };auto f2 = [] { return 1; };         //省略空参数表

        lambda表达式可以通过捕获列表捕获一定范围的变量:

        1、[]不捕获任何变量。

        2、[&]捕获外部作用域中所有变量,并作为引用在函数体中使用(按引用捕获)。

        3、[=]捕获外部作用域中所有变量,并作为副本在函数体中使用(按值捕获)。

        4、[=, &foo]按值捕获外部作用域中所有变量,并按引用捕获foo变量。

        5、[bar]按值捕获bar变量,同时不捕获其他变量。

        6、[this]捕获当前类中的this指针,让lambda表达式拥有和当前类成员函数同样的访问权限。如果已经使用了&或者=,就默认添加此选项。捕获this的目的是可以在lambda中使用当前类的成员函数和成员变量。

lambda表达式基本用法

#include <iostream>
using namespace std;class A
{
public:int i_ = 0;void func(int x, int y){///auto x1 = [] { return _i; };         ///error,没有捕获外部变量auto x2 = [=] { return i_ + x + y; };   ///OK,捕获所有外部变量auto x3 = [&] { return i_ + x + y; };   ///OK,捕获所有外部变量auto x4 = [this] { return i_; };        ///OK,捕获this指针///auto x5 = [this] { return i_ + x + y; }; ///error,没有捕获x,yauto x6 = [this, x, y] { return i_ + x + y; };  ///OK,捕获this指针,x,yauto x7 = [this]{ return i_ ++; };      ///OK,捕获this指针,并修改成员的值}};int main()
{int a = 0, b = 1;auto f1 = [] { return a; };     ///error,没有捕获外部变量auto f2 = [&] { return a++; };  ///OK,捕获所有外部变量,并对a执行自加运算auto f3 = [=] { return a; };    ///OK,捕获所有外部变量,并返回aauto f4 = [=] { return a++; };  ///error,a是以复制方式捕获的,无法修改auto f5 = [a] { return a + b; };    ///error,没有捕获变量bauto f6 = [a, &b] { return a + (b++); };    ///OK,捕获a和b的引用,并对b做自加运算auto f7 = [=, &b] { return a + (b++); };    ///OK,捕获所有外部变量和b的引用,并对b做自加运算return 0;
}

        从上例中可以看到,lambda表达式的捕获列表精细的控制了lambda表达式能够访问的外部变量,以及如何访问这些变量。

        需要注意的是,默认状态下lambda表达式无法修改通过复制方式捕获的外部变量。如果希望修改这些变量的话,我们需要使用引用方式进行捕获。

    int a = 0;auto f = [=] { return a; };a += 1;cout << f() << endl;

        在这个例子中,lambda表达式按值捕获了所有外部变量。在捕获的一瞬间,a的值就已经被复制到f中了。之后a被修改了,但此时f中存储的a仍然还是捕获时的值,因此,最终输出结果是0。

        如果希望lambda表达式在调用时能够及时访问外部变量,我们应当使用引用方式捕获。

        如果希望去修改按值捕获的外部变量,需要显示指明lambda表达式为mutable:

int main()
{int a = 0;///auto f1 = [=](){ return a++; };          ///error,修改按值捕获的外部变量auto f2 = [=]()mutable { return a++; };     ///OK,mutablereturn 0;
}

        需要注意的一点是,被mutable修改的lambda表达式就算没有参数也要写明参数列表。

        lambda表达式的类型在C++11中被称为“闭包类型”。它是一个特殊的,匿名的非nunion的类型。因此,我们可以认为它是一个带有operator()的类,即仿函数。因此,我们可以使用std::function和std::bind来存储和操作lambda表达式:

std::function<int(int)> f1 = [] (int a) { return a; };
std::function<int(void)> f2 = std::bind( [] (int a) { return a; }, 123);

        另外,对于没有捕获任何变量的lambda表达式,还可以被转换成一个普通的函数指针:

using func_t = int(*)(int);
func_t f = [] (int a) { return a; };
f(123);

        lambda表达式可以说是就地定义仿函数闭包的“语法糖”。它的捕获列表捕获住的任何外部变量,最终均会变成闭包类型的成员变量。而一个使用了成员变量的类的operator(),如果能直接被转换为普通的函数指针,那么lambda表达式本身的this指针就会丢失掉。而没有捕获任何外部变量的lambda表达式则不存在这个问题。

        这里也可以很自然的解释为何按值捕获无法修改捕获的外部变量。因为按照C++标志,lambda表达式的operator()默认是const的。一个const成员函数是无法修改成员变量的值的。而mutable的作用,就在于取消operator()的const。

声明式的编程风格,简洁的代码

        就地定义匿名函数,不再需要定义函数对象,大大简化了标准库算法的调用。在C++11之前,我们要调用for_each函数将vector中的偶数打印出来,代码如下所示:

    
#include <iostream>
#include <vector>
#include <functional>
#include <algorithm>
using namespace std;class CountEven
{
public:CountEven(int& c):count(c){}void operator()(int val){if (!(val & 1)){++count;}}private:int& count;
};int main()
{int count = 0;vector<int> v = {1, 2, 3, 4, 5, 6, 7, 8};for_each(v.begin(), v.end(), CountEven(count));cout << "The number of even is " << count << endl;return 0;
}

        这样写既繁琐又容易出错。有了lambda表达式以后,我们可以使用真正的闭包概念来替换这里的仿函数,代码如下所示:

        


#include <iostream>
#include <vector>
#include <functional>
#include <algorithm>
using namespace std;class CountEven
{
public:CountEven(int& c):count(c){}void operator()(int val){if (!(val & 1)){++count;}}private:int& count;
};int main()
{int count = 0;vector<int> v = {1, 2, 3, 4, 5, 6, 7, 8};for_each(v.begin(), v.end(), CountEven(count));cout << "The number of even is " << count << endl;count = 0;for_each(v.begin(), v.end(), [&count](int val){if (!(val & 1)){++ count;}});cout << "The number of even is " << count << endl;return 0;
}

        lambda表达式的价值在于,就地封装短小的功能闭包,可以及其方便的表达出我们希望执行的具体操作,并让上下文结合得更加紧密。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/121480.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

群晖上搭建teamspeak3语音服务器

什么是 TeamSpeak &#xff1f; TeamSpeak &#xff08;简称 TS&#xff09;是一款团队语音通讯工具&#xff0c;但比一般的通讯工具具有更多的功能而且使用方便。它由服务器端程序和客户端程序两部分组成&#xff0c;如果不是想自己架设 TS 服务器&#xff0c;只需下载客户端程…

【vim 学习系列文章 12 -- vimrc 那点事】

文章目录 系统级及本地 vimrc 文件设置 vimrc 的路径 系统级及本地 vimrc 文件 当 Vim 启动时&#xff0c;编辑器会去搜索一个系统级的 vimrc 文件来进行系统范围内的默认初始化工作。 这个文件通常在你系统里 $VIM/vimrc 的路径下&#xff0c;如果没在那里&#xff0c;那你可…

Linux系统编程_网络编程:字节序、socket、serverclient、ftp 云盘

1. 网络编程概述&#xff08;444.1&#xff09; TCP/UDP对比 TCP 面向连接&#xff08;如打电话要先拨号建立连接&#xff09;&#xff1b;UDP 是无连接的&#xff0c;即发送数据之前不需要建立连接TCP 提供可靠的服务。也就是说&#xff0c;通过 TCP 连接传送的数据&#xf…

Android 13.0 SystemUI状态栏屏蔽掉通知栏不显示通知

1.概述 在13.0的系统产品开发中,在SystemUI定制化开发中,有产品需求要求屏蔽通知显示,由于对状态栏的通知管控的比较严,所以要求屏蔽掉通知栏的通知不显示通知 接下来就需要对通知栏的显示流程分析,屏蔽掉通知就可以了 2.SystemUI状态栏屏蔽掉通知栏不显示通知的核心类 f…

【斑梨】世界最小?linux开发板?价格只要39元 Luckfox Pico Mini 超越树莓派PICO ESP32 Arduino

教程地址 幸狐Luckfox Pico RV1103 教程合集 斑梨】世界最小&#xff1f;linux开发板&#xff1f;价格只要39元 Luckfox Pico Mini 超越树莓派PICO ESP32 Arduino

析构函数的相关解释

析构函数&#xff08;Destructor&#xff09;是C中一种特殊的成员函数&#xff0c;用于在对象生命周期结束时执行清理和资源释放操作。每个类都可以有一个析构函数&#xff0c;它的名称与类的名称相同&#xff0c;前面加上一个波浪号&#xff08;~&#xff09;。析构函数通常用…

设计模式——单例模式详解

目录 设计模式类型单例模式单例模式方式饿汉式静态常量方式静态代码块形式 懒汉式线程不安全&#xff08;不推荐&#xff09;懒汉式优化&#xff08;不推荐&#xff09; 双重检查&#xff08;推荐方式&#xff09;静态内部类&#xff08;推荐方式&#xff09;枚举方式&#xff…

STM32 ADC数模转换器

STM32 ADC数模转换器 ADC简介 ADC&#xff08;Analog-Digital Converter&#xff09;模拟-数字转换器 ADC可以将引脚上连续变化的模拟电压转换为内存中存储的数字变量&#xff0c;建立模拟电路到数字电路的桥梁 STM32主要是数字电路&#xff0c;数字电路只有高低电平&#xf…

【torch高级】一种新型的概率学语言pyro(01/2)

一、说明 贝叶斯推理&#xff0c;也就是变分概率模型估计&#xff0c;属于高级概率学模型&#xff0c;极有学习价值&#xff1b;一般来说&#xff0c;配合实际活动学习可能更直观&#xff0c;而pyro是pytorch的概率工具&#xff0c;不同于以往的概率工具&#xff0c;只是集中于…

qt中怎么在鼠标停留的位置上显示该点的坐标位置

需要重写控件的mouseMoveEvent方法。 1、自定义一个QLabel控件&#xff0c;然后重写QLabel的mouseMoveEvent customlabel.h#include <QWidget> #include <QHBoxLayout> #include <QLabel>class CustomLabel : public QLabel {Q_OBJECT public:explicit Cus…

python常见爬虫库以及案例

python常见爬虫库以及案例 一、常见库 以下是一些常见的Python爬虫库&#xff0c;按照一般热门程度的排序&#xff1a; Requests&#xff1a;requests库是非常流行的用于发送HTTP请求的库&#xff0c;因其简洁易用和广泛的社区支持而备受青睐。Beautiful Soup&#xff1a;Be…

PY32F002A系列单片机:高性价比、低功耗,满足多样化应用需求

PY32F002A系列微控制器是一款高性能、低功耗的MCU&#xff0c;它采用32位ARM Cortex-M0内核&#xff0c;最高工作频率达到24MHz&#xff0c;提供了强大的计算能力。此外&#xff0c;PY32F002A拥有最大20Kbytes的flash存储器和3Kbytes的SRAM&#xff0c;为简单的数据处理提供了充…

基于LCC的Buck谐振变换器研究

摘 要 Buck 变换器应用广泛&#xff0c;比如可以为音圈电机、直流电机以及电子设备等提供直流供电电源。更高效率和更小体积的Buck 直流调压电源一直是研究的热点。在我们日常生活中&#xff0c; LLC谐振变换器和Buck谐振变换器随处可见&#xff0c;因为其相比其他变换器而言结…

Python Selenium 之数据驱动测试的实现!

数据驱动模式的测试好处相比普通模式的测试就显而易见了吧&#xff01;使用数据驱动的模式&#xff0c;可以根据业务分解测试数据&#xff0c;只需定义变量&#xff0c;使用外部或者自定义的数据使其参数化&#xff0c;从而避免了使用之前测试脚本中固定的数据。可以将测试脚本…

Unity3D 如何用unity引擎然后用c#语言搭建自己的服务器

Unity3D是一款强大的游戏开发引擎&#xff0c;可以用于创建各种类型的游戏。在游戏开发过程中&#xff0c;经常需要与服务器进行通信来实现一些功能&#xff0c;比如保存和加载游戏数据、实现多人游戏等。本文将介绍如何使用Unity引擎和C#语言搭建自己的服务器&#xff0c;并给…

thrust工程化学习(七)----噪声滤除进阶版

0. 简介 之前我们讲过通过体素化分割&#xff0c;并通过判断这个栅格内的点云数目是否大于阈值。从而来鉴别出噪点。而我们学过最近邻搜索后&#xff0c;我们可以来学习一下更加先进的方法—半径搜索噪声滤除&#xff08;Radius Search Noise Filtering&#xff09;。这是点云…

Redis(05)| 数据结构-哈希表

哈希表是一种保存键值对&#xff08;key-value&#xff09;的数据结构。 哈希表中的每一个 key 都是独一无二的&#xff0c;程序可以根据 key 查找到与之关联的 value&#xff0c;或者通过 key 来更新 value&#xff0c;又或者根据 key 来删除整个 key-value等等。 在讲压缩列表…

报错:Could not resolve host: mirrorlist.centos.org;Unknown error

报错&#xff1a;Could not resolve host: mirrorlist.centos.org;Unknown error 一般是因为网络配置错误导致无法连接外网&#xff0c;我们先尝试ping一下www.baidu.com发现无法ping通。 果然&#xff0c;接下来我们就开始排查吧&#xff01;&#xff01; 1.网络配置查看 打开…

SpringBoot解压zip包,读取每个文件内容

SpringBoot解压zip包&#xff0c;读取每个文件内容 一、运用场景 获取本地压缩包&#xff0c;解压后根据文件名称及类型&#xff0c;对读取的文件内容进行业务处理。 二、POM文件依赖 <!--读取文件--><dependency><groupId>org.apache.poi</groupId&g…

win10下Mariadb绿色版安装步骤

使用绿色版的mariadb数据库管理软件&#xff0c;免费开源&#xff0c;可以用来替换MySQL。首先从mariadb官网下载绿色版本的压缩包。解压后、配置好即可以使用。 把他解压缩到C:\mariadb\之下。打开powershell&#xff1a; Cd c:\mariadb\bin .\mysql_install_db.exe 这一…