c++11 lambda匿名函数

概述:

匿名函数:  就是一个没有名字的函数,和匿名对象类似,其只会在所在行起作用。

那么lambda匿名函数的主要用处是什么呢? 
举个例子:  我们之前使用过sort()排序函数,其可以根据你传入的一组数据,根据某种规则进行排序,规则我们既可以自己指定,也可以使用默认(sort()默认是从小到大排序,如果是C语言的话这个必须得自己指定)。

如果,我们希望自己指定排序规则,C语言的方法,使用函数指针,sort()函数会对其进行回调。c++中可以使用函数对象(当然函数指针也可以用),就是一个类中重写了函数调用运算符()的函数,然后sort内部也可以进行调用。在函数实现中我们可以实现自己的规则,然后返回相应的bool值,供sort()函数排序使用。

但是,上面的情况会出现一种问题,就是无论是函数指针还是函数对象都必须存在一个实际的函数来指定排序的规则。但是,可能有时候我们某种排序规则只在程序中的某一个地方使用,或者只使用一次,而且很有可能不同的位置排序的规则还不一样。

对于一个只使用一次的规则,我们专门为它去设计一个函数,会不会感觉很不方便,(需要写添加声明到头文件,写实际的实现等等),而且如果这样的场景多的话,会导致我们写大量只使用一次的排序,很不方便。

这时候就到啦lambda匿名函数的用武之地了,使用lambda匿名函数,你需要去定义函数,直接在你需要某一规则的时候写一个匿名函数就行,而且其只会在被写的位置有用。这样,对于只使用一次的规则,我们只需要写一个在使用位置使用匿名函数就行,会比较方便。


而且,使用lambda匿名函数还可以临时定义我们需要使用的函数。

 

lambda匿名函数的写法 

[外部变量访问方式] (参数) mutable noexcept -> 返回值类型 {函数体}

 [外部变量访问方式]

[] 告诉编译器我们此处使用的是匿名函数,所以其是不可以省略的。

其作用是指定匿名函数的函数体中对于外部变量的使用方式和导入哪些外部变量,记住是外部的变量并不是参数。

main: 

int nub1;

[](int x){nub1 = 1};   

此处的nub1为外部变量,就是定义在匿名函数之外的变量。

通过[]我们可以指定我们在函数内部可以使用的外部变量,以及使用它们的方式:只读还是只可读可写。

表  [外部变量]的定义方式
外部变量格式功能
[]空方括号表示当前 lambda 匿名函数中不导入任何外部变量。
[=]只有一个 = 等号,表示以值传递的方式导入所有外部变量;
[&]只有一个 & 符号,表示以引用传递的方式导入所有外部变量;
[val1,val2,...]表示以值传递的方式导入 val1、val2 等指定的外部变量,同时多个变量之间没有先后次序;
[&val1,&val2,...]表示以引用传递的方式导入 val1、val2等指定的外部变量,多个变量之间没有前后次序;
[val,&val2,...]以上 2 种方式还可以混合使用,变量之间没有前后次序。
[=,&val1,...]表示除 val1 以引用传递的方式导入外,其它外部变量都以值传递的方式导入。
[this]表示以值传递的方式导入当前的 this 指针。

 

上表中说明了[]的具体使用。

但还还需要注意一下几点: 

  • 如果[]内什么都不写,就表示你是无法使用外部变量的,但是全局变量可以。
  • 对于全局变量,无论方括号内部写什么,在函数内部我们都可以对其进行修改
  • 如果[]内部指定了在匿名函数中可以使用的外部变量,那么其它的外部变量就无法使用了,只能使用指定的,如果使用就会报错。(当然全局变量除外)
  • 上面之所以以引用传递和值传递的形式进行区分,是因为当我们对外部数据进行修改的时候存在两种情况,一种是会影响原数据,一种是不影响原数据只修改传入到参数的拷贝(需要mutable)。
  • 其实,你可以将外部变量也理解为参数传递,会将[]指定的对应外部变量传递到函数内部,供匿名函数使用。


int count = 20;
int main(void) {int num1 = 10;int num2 = 20;[]() {         // 不导入任何的外部变量count;     // 全局变量可以访问               num1;      // error 外部变量都不可以访问num2;};[=]() {             // 以值传递导入全部外部变量count = 10;     // 全局变量可以修改              num1  = 10;     // error 外部变量都不可以修改num2  = 10;};[num1]() {         // 指定导入的外部变量count;         // 全局变量可以访问               num1;          // 导入的外部变量也可以访问num2;          // error num2没有导入不可以访问};std::cin.get();
}

 

(参数) 

和普通函数一样匿名函数也可以传入参数,这样才能用于sort()函数,参数指定的规则和普通函数一样。

如果没有参数我们可以省略(),但是如果后面有mutable和noexcept这些修饰符就不可以省略了,自己指定返回值的时候也不可以省略。 

mutable,noexcept 

  • mutable:  我们在[]中指定为值传递的时候,默认是不可以去修改外部变量的,但是如果我们需要修改,就可以在后面加上mutable关键字,这样你就可以在函数体中修改了。
    注意:  即使你可以修改,因为是值传递,你只是修改了传递到函数内部的外部变量的拷贝,并不会影响外部变量的数据。 
int num1 = 10;
int num2 = 20;[=](int& a, int& b){num1 = 10;};     // error 直接修改值传递的外部变量[=](int& a, int& b)mutable {num1 = 10;};     // 加上mutable就可以了std::cout << num1 << std::endl;              // 但是不会影响外部的值

 

  • noexcept :  就是表示该匿名函数不会抛出异常。

 

->返回值 

此种方式可以用来指定函数的返回值。

在一般情况下是可以省略的因为当你的返回值只有一个或者没有返回值,编译器会自动推导出此匿名函数的返回值

但是,如果你的返回值有多个,那么就需要你指定了

#include <vector>main:
int num1 = 10;[=]() {                                // 返回值只有一个编译器自己进行推导if (num1 > 10) return true;else return false;
};[=](){return {1,2};             // error, 返回多个值,必须自己指定返回类型
};[=]()->std::vector<int>{     // 正确return {1,2}; 
};

 

函数体 

函数的具体功能实现。 

 

lambda匿名函数还可以给其取名字

下面代码中,我们使用auto自动推导出类型,然后将匿名函数赋值给了function,此时function就是这个匿名函数了。(这时候匿名函数在定义的当前行结束的时候,不会失效)


下面,我们可以直接使用function传参,调用函数,其实就是将其看做了一个函数名,然后来调用函数。(这个方式C语言的函数指针是类似的)
 

也是通过这样的方式,我们可以在代码使用的地方定义一个函数。通过给匿名函数取别名可以使其在后续的代码中被调用。

int num1 = 10;
int num2 = 20;auto function = [=](int a, int b) {return num1 + num2 + a + b; };// 使用
function(1, 2);

 

使用lambda匿名函数进行数据排序 

#include <algorithm>    // c++算法库
#include <vector>int main(void) {std::vector<int> v1{ 2,5,4,3,1 };// 指定从大到小排序std::sort(v1.begin(), v1.end(), [](int& a, int& b) {return a > b;});   for (auto& i : v1)  std::cout << i << std::endl;std::cin.get();
}

 

 给匿名函数取别名,让其在后续的代码中可以被调用


int main(void) {std::vector<int> v1{ 2,5,4,3,1 };auto function = [v1]() {for (auto& i : v1)std::cout << i << std::endl;};function();std::cin.get();
}

 

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

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

相关文章

关系数据库标准查询语言-SQL-SQL语言概述

一、SQL(Structured Query Language)语言 1、是高度非过程化的语言 2、关系数据库管理系统(RDBMS)都支持SQL标准 3、具有定义、查询、更新、控制四大功能 4、数据库对象由数据库&#xff08;Database&#xff09;、基本表&#xff08;Table&#xff09;、视图&#xff08;V…

python-doipclient 源文档翻译

python-doipclient 源文档翻译 doipclient 是一个纯 Python 3 的诊断通过互联网&#xff08;DoIP&#xff09;客户端库&#xff0c;可用于通过汽车以太网与现代电子控制单元&#xff08;ECUs&#xff09;通信。它从短期同步客户端的角度实现了 ISO-13400&#xff08;2019 年版…

string经典题目(C++)

文章目录 前言一、最长回文子串1.题目解析2.算法原理3.代码编写 二、字符串相乘1.题目解析2.算法原理3.代码编写 总结 前言 一、最长回文子串 1.题目解析 给你一个字符串 s&#xff0c;找到 s 中最长的回文子串。 示例 1&#xff1a; 输入&#xff1a;s “babad” 输出&am…

Yocto - 变量介绍:BASE_WORKDIR、WORKDIR和D

BASE_WORKDIR 指向所有配方的工作目录根目录。默认值为"${TMPDIR}/work"。 Points to the base of the work directory for all recipes. The default value is “${TMPDIR}/work”. $ bitbake -e <recipe-name>| grep ^BASE_WORKDIR BASE_WORKDIR"/home…

自动化测试-Selenium-元素定位

一.元素定位 因为使用selenium进行自动化测试&#xff0c;元素定位是必不可少的&#xff0c;所以这篇文章用于自动化测试中的selenium中的元素定位法。 1.根据id属性进行定位&#xff08;id是唯一的&#xff09; id定位要求比较高&#xff0c;要求这个元素的id必须是固定且唯…

方差的计算(总体方差与样本方差)

方差是数据集中的各个数据与其均值之间差值的平方的平均值。方差的计算公式如下&#xff1a; 对于总体数据&#xff08;即所有数据&#xff09;&#xff1a; σ 2 1 N ∑ i 1 N ( x i − μ ) 2 \sigma^2 \frac{1}{N} \sum_{i1}^N (x_i - \mu)^2 σ2N1​i1∑N​(xi​−μ)2…

Java的自动装箱和自动拆箱

自动装箱和拆箱在Java开发中的应用与注意事项 在Java开发中&#xff0c;自动装箱&#xff08;Autoboxing&#xff09;和自动拆箱&#xff08;Unboxing&#xff09;是指基本数据类型与其对应的包装类之间的自动转换。这些特性可以使代码更加简洁和易读&#xff0c;但在实际项目…

CANoe-Trace窗口无法解析SOME/IP报文、Demo License激活方式改变

1、Trace窗口无法解析SOME/IP报文 在文章《如何让CANoe或Wireshark自动解析应用层协议》中,我们通过设置指定端口号为SOME/IP报文的方式,可以让CANoe中的Trace窗口对此端口号的报文当成是SOME/IP报文进行解析。 Trace窗口就可以根据传输层端口号对payload数据按照SOME/IP协议…

linuxDNS域名解析

文章目录 DNS 是域名系统的简称正向解析反向解析主从服务器解析bond网卡 DNS 是域名系统的简称 域名和IP地址之间的映射关系 互联网中&#xff0c;IP地址是通信的唯一标识&#xff0c;逻辑地址 访问网站 域名解析的目的就是为了实现&#xff0c;访问域名就等于访问IP地址 …

JS(JavaScript)的引用方式介绍与代码演示

天行健&#xff0c;君子以自强不息&#xff1b;地势坤&#xff0c;君子以厚德载物。 每个人都有惰性&#xff0c;但不断学习是好好生活的根本&#xff0c;共勉&#xff01; 文章均为学习整理笔记&#xff0c;分享记录为主&#xff0c;如有错误请指正&#xff0c;共同学习进步。…

CSS:字数超出容器范围,超出部分省略,变成...

用CSS实现文本超出/溢出容器范围时&#xff0c;超出的部分省略&#xff0c;展示… &#xff08;针对block元素&#xff0c;如果是span&#xff0c;需要添加display&#xff1a;block&#xff1b;&#xff09; 单行文本&#xff1a; .overText{white-space: nowrap;overflow:…

中国同胞进来看看,很多外国人想通过CSDN坑咱们中国人

地址&#xff1a;【诈骗离你我很近】中国同胞进来看看国外诈骗新套路。-CSDN博客

SpringBoot+Vue体育馆管理系统(前后端分离)

技术栈 JavaSpringBootMavenMySQLMyBatisVueShiroElement-UI 角色对应功能 学生管理员 功能截图

Linux安装MySQL教程【带图文命令巨详细】

巨详细Linux安装MySQL 1、查看是否有自带数据库或残留数据库信息1.1检查残留mysql1.2检查并删除残留mysql依赖1.3检查是否自带mariadb库 2、下载所需MySQL版本&#xff0c;上传至系统指定位置2.1创建目录2.2下载MySQL压缩包 3、安装MySQL3.1创建目录3.2解压mysql压缩包3.3安装解…

对硬盘的设想2:纸存,硬指针,软指针

“纸存”是设想中的存储器&#xff0c;它只能改写两次&#xff1a;写一次&#xff0c;再改一次&#xff0c;然后就不能再动了。就像拿着钢笔在纸上写字一样&#xff0c;所以叫纸存。 硬指针P、软指针S S abcd S aPcdPx P aPcdPx S aycd ①一个软指针S&#xff0c;指向数据abcd…

DBeaver无法连接Clickhouse,连接失败

DBeaver默认下载的是0.2.6版本的驱动&#xff0c;但是一直连接失败&#xff1a; 报错提示 解决办法 点击上图中的Open Driver Configuration点击库 - 重置为默认状态在弹出的窗口中修改驱动版本号为0.2.4或者其他版本&#xff08;我没有试用过其他版本&#xff09;&#xff0…

vscode软件上安装 Fitten Code插件及使用

一. 简介 前面几篇文章学习了 Pycharm开发工具上安装 Fitten Code插件&#xff0c;以及 Fitten Code插件的使用。 Fitten Code插件是是一款由非十大模型驱动的 AI 编程助手&#xff0c;它可以自动生成代码&#xff0c;提升开发效率&#xff0c;帮您调试 Bug&#xff0c;节省…

FPGA通过移位相加实现无符号乘法器(参数化,封装成IP可直接调用)

目录 1.前言2.原理3.移位无符号乘法器实现&#xff0c;并参数化 微信公众号获取更多FPGA相关源码&#xff1a; 1.前言 在硬件设计中&#xff0c;乘法器是非常重要的一个器件&#xff0c;乘法器的种类繁多&#xff0c;常见的有并行乘法器、移位相加乘法器和查找表乘法器。 并…

PyQt5中建立柱状图并刷新柱状图数据

使用Qt Designer工具搭建界面时&#xff0c;在适当位置预留一个verticalLayout垂直布局控件&#xff0c;用于放置代码生成的上下两个柱状图。新建柱状图的代码如下&#xff1a; class mainWindow(QMainWindow, Ui_MainWindow):def __init__(self):super(mainWindow, self).__i…

详解 Spark Streaming 的 DStream 对象

一、DStream 的创建 1. 通过 RDD 队列 DStream 在内部实现上是一系列连续的 RDD 来表示。每个 RDD 包含有采集周期内的数据 /** 基本语法&#xff1a;StreamingContext.queueStream(queueOfRDDs: Queue, oneAtATime false) */ object DStreamFromRddQueue {def main(args: Ar…