【C++11】Lambda 表达式:基本使用 和 底层原理

文章目录

  • Lambda 表达式
    • 1. 不考虑捕捉列表
      • 1.1 简单使用介绍
      • 1.2 简单使用举例
    • 2. 捕捉列表 [ ] 和 mutable 关键字
      • 2.1 使用方法
        • 传值捕捉
        • 传引用捕捉
      • 2.2 捕捉方法一览
      • 2.3 使用举例
    • 3. lambda 的底层分析


Lambda 表达式

书写格式:

[capture_list](parameters) mutable -> return_type{statement}

[capture_list]:

  • 捕捉列表,不能省略
  • 固定在 lambda 表达式开始的位置,编译器也是根据 [] 来判断接下来的代码是否为 lambda 函数。
  • 捕捉列表能够捕捉上下文中的变量供 lambda 函数使用。

(parameters):

  • 参数列表。
  • 与普通函数的参数列表一致,如果不需要参数传递,则可以连同 () 一起省略。

mutable:

  • 一个修饰符,取消传值捕捉时值的默认 const 属性(lambda 函数默认是一个 const 类型的,里面的值不可修改)。
  • 另:若使用了 mutable 修饰符,则 (参数列表) 是不可省略掉的,即使是参数为空

return_type:

  • 返回值类型。可以省略,编译器会自动推导。

{statement}:

  • 函数体部分,{} 不能省略,但内容可以为空。
  • 在该函数体内,除了可以使用其参数外,还可以使用所有捕获
    到的变量。

如下是最简单的 lambda 对象,没啥用就是了…

[] {};	

1. 不考虑捕捉列表

1.1 简单使用介绍

比如我们要实现一个两个数相加的函数,用 lambda 表达式就需要写成这样

auto add = [](int x, int y)->int {return x + y; };		
cout << add(1, 2) << endl;
//cout << [](int x, int y)->int {return x + y; }(1, 2) << endl;		// 这样写也能运行,但是我们不这样...
解析:= 后面这一坨整体,代表的是一个 lambda 对象,拿这个对象去构造 add后面就可以用 add 去等价调用函数了

可以看出,lambda 表达式实际上可以理解为 匿名函数,该函数无法直接调用,如果想要直接调用,可借助 auto 将其赋值给一个变量。

需要注意的是:

  • 返回值可以忽略(编译器自动完成推导)
  • 函数体语句多的话,可以按照如下格式写
auto add = [](int x, int y)		// 返回值可以省略,编译器可以自动推导
{								// 函数体语句多的话,可以放下来写return x + y;
};
cout << add2(1, 2) << endl;

1.2 简单使用举例

🌰实现商品各个内容的排序:

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());
}

lambda 写法:

书写格式:[capture - list](parameters) mutable -> return-type{ statement}

int main()
{vector<Goods> v = { { "苹果", 2.1, 5 }, { "香蕉", 3, 4 }, { "橙子", 2.2, 3 }, { "菠萝", 1.5, 4 } };// 价格升序auto priceLess = [](const Goods& g1, const Goods& g2)->bool {return g1._price < g2._price; };sort(v.begin(), v.end(), priceLess);	// 相较于仿函数更好调试// 价格降序sort(v.begin(), v.end(), [](const Goods& g1, const Goods& g2)->bool {return g1._price > g2._price; });									// 这个把断点放头上可能会一直出不来,调试的时候跳到了需要手动取消断点走下一个// 改成比较别的类型也很方便sort(v.begin(), v.end(), [](const Goods& g1, const Goods& g2)->bool {return g1._evaluate < g2._evaluate;});sort(v.begin(), v.end(), [](const Goods& g1, const Goods& g2)->bool {return g1._evaluate > g2._evaluate;});return 0;
}

2. 捕捉列表 [ ] 和 mutable 关键字


📕注意事项(具体论证见下文):

  1. 父作用域指包含lambda函数的语句块。

  2. 语法上捕捉列表可由多个捕捉项组成,并以逗号分割。

  3. 捕捉列表不允许变量重复传递,否则就会导致编译错误。

    eg:[=, a]
    = 已经以值传递方式捕捉了所有变量,捕捉 a 重复
    
  4. 在块作用域中的 lambda 函数仅能捕捉父作用域中局部变量,捕捉任何非此作用域或者非局部变量都会导致编译报错。同时有,在块作用域以外的 lambda 函数捕捉列表必须为空。

  5. lambda表达式之间不能相互赋值,即使看起来类型相同


若不使用 lambda 捕捉,实现一个 swap 接口如下:

int x = 0, y = 1;
auto swap1 = [](int& rx, int& ry)
{int tmp = rx;rx = ry;ry = tmp;
};swap1(x, y);
cout << x << " "<< y << endl;

而 捕捉列表 描述了:

  • 上下文中哪些数据可以被 lambda 使用
  • 以及使用的方式 传值 还是 传引用

2.1 使用方法

传值捕捉
  • [val]传值捕捉 [x, y],相当于把 x 和 y “捕捉” 到 lambda 表达式中,直接就可以访问了。这时 lambda 是 const 函数,其中的 x 和 y 不能修改;

  • 加上关键字 mutable 就可以修改了,不过此时的 x 和 y 就是函数形参。

  • 另外需要注意的是,有 mutable 时,参数列表的括号不能省略。建议平时也不要省略

并不能达到效果的错误使用:

// err,这是一个错误的写法
int x = 0, y = 1;
auto swap2 = [x, y]() mutable 
{int tmp = x;x = y;y = tmp;
};
swap2();
cout << x << " " << y << endl;

像如上,虽然对 x 和 y 进行了捕捉,也加上了 mutable 使其可以修改,但实际并达不到我们让全局变量 x 和 y 修改的效果。因为 lambda 中他们只是形参,一份临时拷贝的对象。

传引用捕捉
  • [&val] 是将外面的值传引用到 lambda 内部
  • 需要在 lambda 内部对某个变量修改时用传引用捕捉

真正修改了外面的参数:

// 这里的 &x 就是引用捕捉int& x(不是取地址
// 引用捕捉
int x = 0, y = 1;
auto swap2 = [&x, &y]()
{int tmp = x;x = y;y = tmp;
};
swap2();
cout << x << " " << y << endl;

2.2 捕捉方法一览

混合捕捉

[var]:表示 值传递方式 捕捉变量 var
[this]:表示 值传递方式 捕捉当前的 this 指针
[&var]:表示 引用传递捕捉 变量 var

auto func1 = [&x, y]()
{//...
};

全部引用捕捉

[&]:表示 引用传递捕捉 所有父作用域中的变量(包括 this)

auto func2 = [&]()
{//...
};

全部传值捕捉

[=]:表示值传递方式捕获所有父作用域中的变量(包括 this)

auto func3 = [=]()
{//...
};

全部引用捕捉,x 传值捕捉

auto func4 = [&, x]()
{//...
};

此外排列组合:

[=, &a, &b]:以 引用传递 的方式捕捉变量 a 和 b,值传递方式 捕捉其他所有变量

[&,a, this]:以 值传递方式 捕捉变量 a 和 this,引用方式 捕捉其他变量


2.3 使用举例

先来个讲解前提,创建线程:

  • Linux 下创建线程:pthread_create(posix)
  • C++98,linux 和 windows 下都可以支持的多线程程序:条件编译。
#ifdef _WIN32CreateThread
#elsepthread_create
#endif 
  • C++11,linux 和 windows 下都可以支持的多线程程序:thread库。

🌰要求 m 个线程分别打印 1~n

线程的传统写法:

void Func1(int n, int num)
{for (int i = 0; i < n; i++){cout <<num<<":" << i << endl;}cout << endl;
}
int main()
{int n1, n2;cin >> n1 >> n2;thread t1(Func1, n1, 1);thread t2(Func1, n2, 2);t1.join();t2.join();return 0;
}

lambda 写法:第一种,较为冗余,不便于添加线程

int main()		// 这个版本蛮冗余
{int n1, n2;cin >> n1 >> n2;thread t1([n1](int num){for (int i = 0; i < n1; i++){cout <<num<<":" << i << endl;}cout << endl;}, 1);thread t2([n2](int num){for (int i = 0; i < n2; i++){cout << num << ":" << i << endl;}cout << endl;}, 2);t1.join();t2.join();return 0;
}

lambda 写法:第二种,推荐

int main()
{size_t m;cin >> m;vector<thread> vthds(m);// 要求 m 个线程分别打印 1~nfor (size_t i = 0; i < m; i++){size_t n;cin >> n;vthds[i] = thread([i, n, m]() {		// 匿名的lambda对象,移动赋值给的vhds[i]for (int j = 0; j < n; j++){cout << i << ":" << j << endl;}cout << endl;});}for (auto& t : vthds)	// thread 不支持拷贝构造(delete了),这里要加引用才跑得动{t.join();}return 0;
}

3. lambda 的底层分析

先说结论,实际在底层编译器对于 lambda 表达式的处理方式,完全就是按照函数对象(仿函数)的方式处理的。

即:如果定义了一个 lambda 表达式,编译器会自动生成一个类,在该类中重载了 operator()。

  • 参数列表 会变成 仿函数的参数
  • 函数体 就是 仿函数主体
  • lambda 对象的类型 就是 仿函数的类型(见后文)
  • 如下这个仿函数类是一个没有给成员变量的空类,所以大小是 1 个字节 反汇编可以查到
int x = 0, y = 1;
int m = 0, n = 1;auto swap1 = [](int& rx, int& ry)
{int tmp = rx;rx = ry;ry = tmp;
};cout << sizeof(swap1) << endl;	// 输出 1 

有如下一个模拟计算理财收益的类:

class Rate
{
public:Rate(double rate) : _rate(rate)	{}			// 传入利率double operator()(double money, int year)	// 参数:本金和年限,返回收益{return money * _rate * year;			// 模拟计算}
private:double _rate;
};
  • 以下代码,函数对象的汇编过程:call 仿函数的构造函数,再 call operator()
// 函数对象
int main()
{double rate = 0.49;Rate r1(rate);r1(10000, 2);cout << sizeof(r1) << endl;	// 8return 0;
}
  • 以下代码,lambda 汇编过程:call lambda_uuid 类的构造函数,再call <lambda_uuid>::operator()

  • uuid 是一个电脑生成的唯一随机值,作为 lambda 类的后缀,刚刚好唯一标识

查看 r2 的大小,r2 里面没使用 lambda 涉及参数之外的 变量 n(如果用了的话根据内存对齐规则,r2 的大小是 16),也就是说:

  • [=] 实际使用的值,才会在 lambda 类中作为成员变量初始化,所以这里的 8 是 double _rate 的大小
// lambda
int main()
{		double rate = 0.49;int n = 0;	// 测试 [=] 全部传值捕捉auto r2 = [=](double money, int year)->double {return money * rate * year; };r2(10000, 2);cout << sizeof(r2) << endl;	// 8return 0;
}// 所以:lambda 和范围 for 一样,底层是用别的实现的,被封装了一趟而已
  • 之前提及的 lambda_uuid,才是 lambda 表达式底层的类型名称,编译器会给他们唯一标识的名称

下面代码,f1 和 f2 即使看着一样,如上阐述,实则底层的类型名称都不同,不能互相赋值:

auto f1 = [] {cout << "hello world" << endl; };
auto f2 = [] {cout << "hello world" << endl; };
//f1 = f2;	// err...

🥰如果本文对你有些帮助,请给个赞或收藏,你的支持是对作者大大莫大的鼓励!!(✿◡‿◡) 欢迎评论留言~~


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

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

相关文章

C++(STL容器适配器)

前言&#xff1a; 适配器也称配接器&#xff08;adapters&#xff09;在STL组件的灵活组合运用功能上&#xff0c;扮演着轴承、转换器的角色。 《Design Patterns》对adapter的定义如下&#xff1a;将一个class的接口转换为另一个class的接口&#xff0c;使原本因接口不兼容而…

信息学奥赛一本通-编程启蒙3318:练54.1 6084问题

3318&#xff1a;练54.1 6084问题 时间限制: 1000 ms 内存限制: 65536 KB 提交数: 419 通过数: 300 从这里进入题目&#xff1a;信息学奥赛一本通-编程启蒙&#xff08;C版&#xff09;在线评测系统 【题目描述】 任意给出一个四位数&#xff0c;把它重新组成一个…

【C语言】语法--联合体union详解

本文参考博客: https://blog.csdn.net/m0_57180439/article/details/120417270 定义及示例: 联合是一种特殊的自定义类型,该种类型定义的变量也包含一系列的成员,特征是这些成员共用同一块空间,所以联合体也被称为共用体。 #include<stdio.h> union Un//联合类型…

Eureka

大家好我是苏麟今天带来Eureka的使用 . 提供者和消费者 在服务调用关系中&#xff0c;会有两个不同的角色&#xff1a; 服务提供者&#xff1a;一次业务中&#xff0c;被其它微服务调用的服务。&#xff08;提供接口给其它微服务&#xff09; 服务消费者&#xff1a;一次业务…

CCF CSP认证 历年题目自练 Day22

CCF CSP认证 历年题目自练 Day22 题目一 试题编号&#xff1a; 201912-1 试题名称&#xff1a; 报数 时间限制&#xff1a; 1.0s 内存限制&#xff1a; 512.0MB 题目分析&#xff08;个人理解&#xff09; 每一个人都要报多少个数字&#xff0c;我选择字典存储&#xff0…

私有云盘:lamp部署nextcloud+高可用集群

目录 一、实验准备&#xff1a; 二、配置mariadb主从复制 三台主机下载mariadb 1&#xff09;主的操作 2&#xff09;从的操作 3&#xff09;测试数据是否同步 三、配置nfs让web服务挂载 1、安装 2、配置nfs服务器 3、配置web服务的httpd 4、测试 四、web 服务器 配…

五种I/O模型

目录 1、阻塞IO模型2、非阻塞IO模型3、IO多路复用模型4、信号驱动IO模型5、异步IO模型总结 blockingIO - 阻塞IOnonblockingIO - 非阻塞IOIOmultiplexing - IO多路复用signaldrivenIO - 信号驱动IOasynchronousIO - 异步IO 5种模型的前4种模型为同步IO&#xff0c;只有异步IO模…

【m_listCtrl !=NULL有多个运算符与操作数匹配】2023/9/21 上午11:03:44

2023/9/21 上午11:03:44 m_listCtrl !=NULL有多个运算符与操作数匹配 2023/9/21 上午11:04:00 如果您在编译或运行代码时遇到"M_listCtrl != NULL有多个运算符与操作数匹配"的错误提示,这通常是由于以下几个原因之一: 错误使用运算符:在条件判断语句中,应该使…

进程调度算法之时间片轮转调度(RR),优先级调度以及多级反馈队列调度

1.时间片轮转调度算法(RR) round Robin 1.算法思想 公平地、轮流地为各个进程服务&#xff0c;让每个进程在一定时间间隔内都可以得到响应。 2.算法规则 按照各进程到达就绪队列的顺序&#xff0c;轮流让各个进程执行一个时间片&#xff08;如100ms&#xff09;。 若进程未…

C/C++学习 -- HMAC算法

1. HMAC算法概述 HMAC&#xff0c;全称为HMAC-MD5、HMAC-SHA1、HMAC-SHA256等&#xff0c;是一种在数据传输中验证完整性和认证来源的方法。它结合了哈希函数和密钥&#xff0c;通过在数据上应用哈希函数&#xff0c;生成一个带密钥的散列值&#xff0c;用于验证数据的完整性。…

Kafka 搭建过程

目录 1.关于Kafka2.Kafka 搭建过程3.参考 本文主要介绍Kafka基本原理&#xff0c;以及搭建过程。 1.关于Kafka Apache Kafka是一个开源的分布式事件流平台&#xff0c;被设计用来实现实时数据流的发布、订阅、存储和处理。 Kafka的主要特性包括&#xff1a; 高吞吐量&#x…

10.1 今日任务:select实现服务器并发

#include <myhead.h>#define ERR_MSG(msg) do{\fprintf(stderr, "__%d__:", __LINE__); \perror(msg);\ }while(0)#define PORT 8888 //端口号&#xff0c;范围1024~49151 #define IP "192.168.112.115" //本机IP&#xff0c;ifco…

【Vue3】定义全局变量和全局函数

// main.ts import { createApp } from vue import App from ./App.vue const app createApp(App)// 解决 ts 报错 type Filter {format<T>(str: T): string } declare module vue {export interface ComponentCustomProperties {$filters: Filter,$myArgs: string} }a…

ubuntu安装MySQL

一行指令即可! sudo apt install mysql-server常用MySQL服务指令 sudo service mysql status # 查看服务状态 sudo service mysql start # 启动服务 sudo service mysql stop # 停止服务 sudo service mysql restart # 重启服务终端里面进入Mysql 其中-u后面root是我的用户名…

计算机竞赛 题目:基于FP-Growth的新闻挖掘算法系统的设计与实现

文章目录 0 前言1 项目背景2 算法架构3 FP-Growth算法原理3.1 FP树3.2 算法过程3.3 算法实现3.3.1 构建FP树 3.4 从FP树中挖掘频繁项集 4 系统设计展示5 最后 0 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 基于FP-Growth的新闻挖掘算法系统的设计与实现…

httpserver 下载服务器demo 以及libevent版本的 httpserver

实现效果如下&#xff1a; 图片可以直接显示 cpp h 这些可以直接显示 其他的 则是提示是否要下载 单线程 还有bug 代码如下 先放上来 #include "httpserver.h" #include "stdio.h" #include <stdlib.h> #include <arpa/inet.h> #include…

QScrollArea样式

简介 QScrollBar垂直滚动条分为sub-line、add-line、add-page、sub-page、up-arrow、down-arrow和handle几个部分。 QScrollBar水平滚动条分为sub-line、add-line、add-page、sub-page、left-arrow、right-arrow和handle几个部分。 部件如下图所示&#xff1a; 样式详…

数据结构与算法(一):概述与复杂度分析

参考引用 Hello 算法 Github 仓库&#xff1a;hello-algo 1. 初识算法 1.1 算法无处不在 1.1.1 二分查找&#xff1a;查阅字典 在字典里&#xff0c;每个汉字都对应一个拼音&#xff0c;而字典是按照拼音字母顺序排列的。假设我们需要查找一个拼音首字母为 r 的字&#xff0…

美妆护肤品商城小程序的作用是什么?

化妆品几乎可以覆盖所有人群&#xff0c;各式各样的品牌及经销商非常多&#xff0c;主要销售模式为门店零售、线上入驻电商平台售卖、批发等&#xff0c;近些年随着电商发展迭代以及消费升级&#xff0c;对品牌或经销商来说&#xff0c;传统经营模式变得低效&#xff0c;每个人…

小程序如何使用自定义组件

使用自定义组件的步骤如下&#xff1a; 创建自定义组件&#xff1a;在小程序项目根目录下的 components 文件夹中创建一个文件夹&#xff0c;然后在该文件夹中创建一个 .json 文件、一个 .wxml 文件和一个 .js 文件&#xff0c;这三个文件分别对应组件的配置、模板和逻辑。 在…