【C++11】(lambda)

C++11中的lambda与线程。

目录

  • Lambda:
    • 仿函数的缺点:
    • Lambda语法:
    • Lambda使用示例:
      • 两数相加:
      • 两数交换:
      • 解决Goods排序问题:
    • Lambda原理:

Lambda:

假设我们有一个商品类,需要对指定的元素进行排序。

struct Goods
{string _name; double _price; // 价格int _evaluate; // 评价Goods(const char* str, double price, int evaluate): _name(str), _price(price), _evaluate(evaluate){}
};

仿函数的缺点:

首先我们会想到可以使用仿函数进行排序。

struct ComparePriceLess
{bool operator()(const Goods& gl, const Goods& gr){return gl._price < gr._price;}
};
struct ComparePriceGreater
{bool operator()(const Goods& gl, const Goods& gr){return gl._price > gr._price;}
};
int main()
{vector<Goods> v = { { "苹果", 2.1, 5 }, { "香蕉", 3, 4 }, { "橙子", 2.2, 3 }, { "菠萝", 1.5, 4 } };sort(v.begin(), v.end(), ComparePriceLess());sort(v.begin(), v.end(), ComparePriceGreater());return 0;
}

随着C++语法的发展,人们开始觉得上面的写法太复杂了,每次为了实现一个比较算法,都要重新去写一个类,如果每次比较的逻辑不一样,还要去实现多个类,特别命令更是关键,最好是可以清晰反映该仿函数功能的。

而且,如果该仿函数是定义在别的文件且命名不规范,打包为动态库我们无法得知该仿函数的功能。

这些都给编程者带来了极大的不便。因此,在C++11语法中出现了Lambda表达式。

我们可以将Lambda看作为函数内定义的函数。方便接受理解。

Lambda语法:

lambda表达式书写格式:

[capture-list] (parameters) mutable -> return-type { statement }
  • [capture-list] : 捕捉列表,该列表总是出现在lambda函数的开始位置,编译器根据[]来判断接下来的代码是否为lambda函数,捕捉列表能够捕捉上下文中的变量供lambda函数使用。
  • (parameters):参数列表。与普通函数的参数列表一致,如果不需要参数传递,则可以连同()一起省略
  • mutable:默认情况下,lambda函数总是一个const函数,mutable可以取消其常量性。使用该修饰符时,参数列表不可省略(即使参数为空)。
  • ->returntype:返回值类型。用追踪返回类型形式声明函数的返回值类型,没有返回值时此部分可省略。返回值类型明确情况下,也可省略,由编译器对返回类型进行推导。
  • {statement}:函数体。在该函数体内,除了可以使用其参数外,还可以使用所有捕获到的变量。

注意:

在lambda函数定义中,参数列表和返回值类型都是可选部分,而捕捉列表和函数体可以为空。因此C++11中最简单的lambda函数为:[]{}; 该lambda函数不能做任何事情。

Lambda使用示例:

有了usage当然要看一看具体如何使用啦。

两数相加:

auto add = [](int x, int y)->int { return x + y; };
int ret = add(1, 5);
cout << ret << endl;

和普通函数很像:
返回值(->int),参数(int x, int y),函数体 { return x + y; }

两数交换:

我们先来看如下的lambda语句是否正确?

auto swap = [](int x, int y)-> void
{int tmp = x;x = y;y = tmp;
};

显然是错误的,实际上我们的x与y也就是普通的传值
在这里插入图片描述

因此我们传引用即可解决。
在这里插入图片描述
那么捕捉列表是干啥用?
在这之前我们要知道lambda的作用域与当前函数的域不是父域子域的关系,而是独立的一个域,也就是说lambda是用不了外部域的对象的。

但是我们可以进行捕捉进而使用外部作用域对象。
即在捕捉列表中进行输入你想捕捉的对象即可。
在这里插入图片描述
但是这样会报错,原因在于我们默认捕捉得到的参数是自带const的值复制捕捉。我们在前面的语法也提到过,因此我们可以加上mutable即可去掉const属性。

结果:
在这里插入图片描述
仍旧没有成功交换,是因为我们的捕捉得到的是传值的,所以这也侧面说明其实mutable的作用不是很大,我们也不常用,只有在特定情境下才有用处。

所以我们在a,b前加上&即可引用。
在这里插入图片描述
另外,如果我们需要捕捉的参数过多,也可以全部引用捕捉,或者值复制捕捉,或者混合捕捉,十分灵活。

[&]() {}; // 全部引用捕捉
[=]() {}; // 全部传值捕捉
[&, a]() {};  // 全部引用捕捉,a是传值捕捉
[=, &a]() {}; // 全部传值捕捉,a是引用捕捉

解决Goods排序问题:

int main()
{vector<Goods> v = { { "苹果", 2.1, 5 }, { "香蕉", 3, 4 }, { "橙子", 2.2, 3 }, { "菠萝", 1.5, 4 } };sort(v.begin(), v.end(), [](const Goods& g1, const Goods& g2) {return g1._price < g2._price; });sort(v.begin(), v.end(), [](const Goods& g1, const Goods& g2) {return g1._price > g2._price; });sort(v.begin(), v.end(), [](const Goods& g1, const Goods& g2){return g1._evaluate < g2._evaluate; });sort(v.begin(), v.end(), [](const Goods& g1, const Goods& g2) {return g1._evaluate > g2._evaluate; });
}

结论:
捕捉列表描述了上下文中那些数据可以被lambda使用,以及使用的方式传值还是传引用。
[var]:表示值传递方式捕捉变量var
[=]:表示值传递方式捕获所有父作用域中的变量(包括this)
[&var]:表示引用传递捕捉变量var
[&]:表示引用传递捕捉所有父作用域中的变量(包括this)
[this]:表示值传递方式捕捉当前的this指针

注意:
a. 父作用域指包含lambda函数的语句块
b. 语法上捕捉列表可由多个捕捉项组成,并以逗号分割。
比如:[=, &a, &b]:以引用传递的方式捕捉变量a和b,值传递方式捕捉其他所有变量[&,a, this]:值传递方式捕捉变量a和this,引用方式捕捉其他变量
c. 捕捉列表不允许变量重复传递,否则就会导致编译错误。
比如:[=, a]:=已经以值传递方式捕捉了所有变量,捕捉a重复
d. 在块作用域以外的lambda函数捕捉列表必须为空。
e. lambda表达式之间不能相互赋值,即使看起来类型相同

void (*PF)();
int main()
{auto f1 = [] {cout << "hello world" << endl; };auto f2 = [] {cout << "hello world" << endl; };// 此处先不解释原因,等lambda表达式底层实现原理看完后,大家就清楚了//f1 = f2;  // 编译失败--->提示找不到operator=()// 允许使用一个lambda表达式拷贝构造一个新的副本auto f3(f2);f3();// 可以将lambda表达式赋值给相同类型的函数指针PF = f2;PF();return 0;
}

Lambda原理:

我们的lambda对象是多大?类型是什么?为什么不能相互赋值?
这就涉及到原理了。
我们看一下汇编即可清楚的得到结论:

class Rate
{
public:Rate(double rate) : _rate(rate){}double operator()(double money, int year){return money * _rate * year;}
private:double _rate;
};
int main()
{// 函数对象double rate = 0.49;Rate r1(rate);r1(10000, 2);// lambdaauto r2 = [=](double monny, int year)->double {return monny * rate * year;};r2(10000, 2);return 0;
}

函数对象将rate作为其成员变量,在定义对象时给出初始值即可,lambda表达式通过捕获列表可以直接将该变量捕获到。
在这里插入图片描述
因此我们得到结论,lambda底层就是仿函数,他的类名为lambda_uuid,uuid是一个几乎不会发生重复的数字生成器,所以对象大小为1,类型不同不能相互赋值。

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

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

相关文章

山东航空小程序查询

山东航空小程序 1) 请求地址 https://scxcx.sda.cn/mohe/proxy?url/trp/ticket/search 2) 调用方式&#xff1a;HTTP post 3) 接口描述&#xff1a; 接口描述详情 4) 请求参数: {"dep": "TAO","arr": "HRB","flightDate&qu…

ClickHouse集成LDAP实现简单的用户认证

1.这里我的ldap安装的是docker版的 docker安装的化就yum就好了 sudo yum install docker-ce docker-ce-cli containerd.io docker-buildx-plugin docker-compose-plugin sudo systemctl start docker 使用下面的命令验证sudo docker run hello-world docker pull osixia/openl…

微信小程序与本地MySQL数据库通信

微信小程序与本地MySQL数据库通信 因为本地MySQL服务器没有域名&#xff0c;也没有进行相应的请求操作封装&#xff0c;因此微信小程序没办法和数据库通信。 但是对于开发人员来说&#xff0c;没有数据库&#xff0c;那还能干撒&#xff1f;虽然我尝试过用json-server&#x…

好用的AI搜索引擎

1. 360AI 搜索 访问 360AI 搜索: https://www.huntagi.com/sites/1706642948656.html 360AI 搜索介绍&#xff1a; 360AI 搜索&#xff0c;新一代智能答案引擎&#xff0c;值得信赖的智能搜索伙伴&#xff0c;为复杂搜索提供专业支持&#xff0c;解锁更相关、更全面的答案。AI…

数据库day2

复盘上一天的内容&#xff1a; 数据库的种类&#xff1f;你了解的数据库?什么是数据库&#xff1f; 怎么增加表中数据 &#xff0c;怎么删除表中数据&#xff0c; 怎么创建表中数据。 数据库编程的操作步骤&#xff1f; 打开数据库--增删改查--关闭数据库。 s…

探索Facebook的最新更新:社交体验的新高度

Facebook作为全球领先的社交媒体平台&#xff0c;一直致力于不断创新和改进&#xff0c;以提供更优质的用户体验。近期&#xff0c;Facebook推出了一系列新的更新&#xff0c;旨在提升用户的社交互动体验和平台功能。本文将详细探讨这些最新更新&#xff0c;分析其对用户和社交…

【cocos creator】【TS】ts入门,cocos creator基础知识

TS入门 记录常用的 系统学习可参考&#xff1a;菜鸟教程 1、语法规范&#xff1a; 以换行分割语句&#xff0c; 可省略末尾分号&#xff0c;同一行需要使用分号来分隔 2.打印log console.log("Hello World!")//白色&#xff0c;一般提示 console.warn("Hel…

如何用AI交互数字人一体机,打造政务服务新名片?

如今&#xff0c;将“高效办成一件事”作为优化政务服务、提升行政效能的重要抓手&#xff0c;各地方为了促进政务服务由传统模式向数字化、智能化方向转变&#xff0c;纷纷在政务服务场景融合了AI交互数字人&#xff0c;实现“无人化、智慧化”导办、帮办、代办等模式&#xf…

9、Redis之哨兵

Redis之哨兵 1、什么是哨兵 ​ 对于Master宕机后的冷处理方式是无法实现高可用的。Redis从2.6版本开始提供了高可用的解决方案–Sentinel哨兵机制。在集群中再引入一个节点&#xff0c;该节点充当Sentinel哨兵&#xff0c;用于监视Master的运行状态&#xff0c;并在Master宕机…

Gemma的简单理解;Vertex AI的简单理解,与chatGpt区别

目录 Gemma的简单理解 Vertex AI的简单理解 Gemma Vertex AI Gemma Vertex AI和chatcpt区别 一、定义与功能 二、技术特点 三、应用场景 四、获取与部署 Gemma的简单理解 定义与功能: Gemma是谷歌开源的一款大语言模型,它采用了Gemini架构,并提供了20亿(2B)和7…

Apache AGE的MATCH子句

MATCH子句允许您在数据库中指定查询将搜索的模式。这是检索数据以在查询中使用的主要方法。 通常在MATCH子句之后会跟随一个WHERE子句&#xff0c;以添加用户定义的限制条件到匹配的模式中&#xff0c;以操纵返回的数据集。谓词是模式描述的一部分&#xff0c;不应被视为仅在匹…

3D问界—在MAYA中使用Python脚本进行批量轴居中

问题提出&#xff1a;MAYA中如何使用Python脚本 今天不是一篇纯理论&#xff0c;主要讲一下MAYA中如何使用Python脚本&#xff0c;并解决一个实际问题&#xff0c;文章会放上我自己的代码&#xff0c;若感兴趣欢迎尝试&#xff0c;当然&#xff0c;若有问题可以见文章末尾渠道&…

分布式IO系统2通道串口通信模块M602x

现场总线耦合器本身包含一个电源模块&#xff0c;它有 2 个串口通道&#xff0c;通过 Modbus RTU&#xff08;Master&#xff09;协议连接外部串行设备&#xff0c;实现耦合器与外部串行设备通信&#xff0c;现以连接设备的示例带大家了解我们钡铼的2 通道串口通信模块 M602x。…

【BUG】已解决:Uncaught SyntaxError: Unexpected token ‘<‘

已解决&#xff1a;Could not install packages due to an EnvironmentError: [Errno 13] Permission denied 欢迎来到我的主页&#xff0c;我是博主英杰&#xff0c;211科班出身&#xff0c;就职于医疗科技公司&#xff0c;热衷分享知识&#xff0c;武汉城市开发者社区主理人 …

深入解析Apache Hive架构

目录 引言Hive简介 什么是HiveHive的特性Hive的优势 Hive架构概述 Hive的核心组件Hive的工作原理 Hive的核心组件详解 HiveQLMetaStoreDriverCompilerOptimizerExecutorStorage Hive的工作流程 查询处理流程数据加载和存储流程 Hive的使用场景Hive的优缺点总结 引言 随着大数…

【INTEL(ALTERA)】使用 F-Tile DisplayPort FPGA IP 设计示例时为何模拟失败?

目录 说明 解决方法 说明 由于 Quartus Prime Pro Edition 软件版本 24.1 存在一个问题&#xff0c;F-Tile DisplayPort FPGA IP 设计示例的模拟将运行超过 24 小时&#xff0c;然后失败&#xff0c;并显示消息“Simulation Hanged”。 解决方法 要解决此问题&#xff0c;…

Neither the JAVA_HOME nor the JRE_HOME environment variable is defined问题解决

一、系统环境变量中添加tomcatjdk的环境变量声明 1、右击此电脑->属性->高级系统设置 可复制粘贴下面的变量名 CATALINA_HOME 点击path->编辑->新建 可将下面值粘入 %CATALINA_HOME%\bin 2、配置jdk的系统变量 系统变量->新建->如图 可将下面变量名粘入 J…

Python兴趣编程百例:使用Python实现一个Json与Excel互转小工具

在日常数据处理中,JSON 和 Excel 是两种常见的数据格式。为了方便在这两种格式之间进行转换,我们决定使用 Python 的 pandas 库来实现 JSON 与 Excel 的互相转换工具,并用 PyQt 实现一个图形界面。本项目将包含详细的设计思路、类设计说明、实现步骤和完整代码。 1.设计思路…

Flutter热更新技术探索

一&#xff0c;需求背景&#xff1a; APP 发布到市场后&#xff0c;难免会遇到严重的 BUG 阻碍用户使用&#xff0c;因此有在不发布新版本 APP 的情况下使用热更新技术立即修复 BUG 需求。原生 APP&#xff08;例如&#xff1a;Android & IOS&#xff09;的热更新需求已经…

c++写一个查询工作站服务器中每个用户的空间使用量

工作中有一个小需求&#xff0c;需要统计用户空间使用量 统计空间的主函数&#xff1a; #include <iostream> #include <fstream> #include <sstream> #include <map> #include <string> #include <cstdlib> #include <sys/types.h&g…